|
|
# encoding=utf-8
|
|
|
class GamesService
|
|
|
include ApplicationHelper
|
|
|
include GamesHelper
|
|
|
CODES = %W(1 2 3 4 5 6 7 8 9 A B C D E F G H J K L N M O P Q R S T U V W X Y Z)
|
|
|
|
|
|
# 关卡详情
|
|
|
def show_game params, current_user
|
|
|
game = Game.min.find_by_identifier(params[:identifier])
|
|
|
Rails.logger.info("game id is #{game.try(:id)}, current_user id is #{current_user.try(:id)}")
|
|
|
if game.blank? || current_user.blank?
|
|
|
return{:status => 404}
|
|
|
end
|
|
|
# 防止评测中途ajax被取消;3改成0是为了处理首次进入下一关的问题
|
|
|
game.update_attribute(:status, 0) if game.status == 1
|
|
|
game.update_attributes(:status => 0, :open_time => Time.now) if game.status == 3 # Redo:后续可以去掉,只是为了兼顾其它模块的耗时统计
|
|
|
|
|
|
myshixun = Myshixun.min.find(game.myshixun_id)
|
|
|
shixun = Shixun.min.find(myshixun.shixun_id)
|
|
|
unless (myshixun.user_id == current_user.id || current_user.admin? || current_user.id == shixun.try(:user_id) || current_user.is_certification_teacher)
|
|
|
return{:status => 403}
|
|
|
end
|
|
|
game_challenge = Challenge.min.find(game.challenge_id)
|
|
|
unless current_user.is_certification_teacher || current_user.admin?
|
|
|
game_challenge.answer = ""
|
|
|
end
|
|
|
|
|
|
has_answer = game_challenge.answer.blank? ? false : true
|
|
|
game_challenge.answer = nil
|
|
|
if game_challenge.st == 1
|
|
|
game_challenge.score = game_challenge.choose_score.to_i
|
|
|
end
|
|
|
# st:判断是选择类型还是实训类型
|
|
|
st = game_challenge.st
|
|
|
game_count = myshixun.games.count
|
|
|
discusses_count = shixun.discusses.count
|
|
|
mirror_name = myshixun.mirror_name
|
|
|
user = myshixun.owner
|
|
|
username = user.show_name
|
|
|
image_url = url_to_avatar(user)
|
|
|
user_url = "/users/#{user.login}"
|
|
|
is_teacher = (user.user_extensions.identity == 0)
|
|
|
tpm_identifier = shixun.try(:identifier)
|
|
|
# 实训超时设置
|
|
|
time_limit = shixun.exec_time
|
|
|
|
|
|
# 高性能取上一关、下一关
|
|
|
prev_game = Game.prev_identifier(shixun.id, game.myshixun_id, game_challenge.position)
|
|
|
next_game = if current_user.is_certification_teacher || shixun_manager(shixun, current_user) || game.status || shixun.task_pass
|
|
|
Game.next_game(shixun.id, game.myshixun_id, game_challenge.position).try(:identifier)
|
|
|
end
|
|
|
|
|
|
# 该参数是为了判断是否需要加载loading
|
|
|
@is_subject = params[:is_subject]
|
|
|
|
|
|
# 关卡点赞数, praise_or_tread = 1则表示赞过
|
|
|
praise_count = PraiseTread.where(:praise_tread_object_id => game_challenge.id, :praise_tread_object_type => "Challenge", :praise_or_tread => 1).count
|
|
|
user_praise = PraiseTread.where(:praise_tread_object_id => game_challenge.id, :praise_tread_object_type => "Challenge", :user_id => current_user.id, :praise_or_tread => 1).present? ? true : false
|
|
|
|
|
|
# 实训的最大评测次数,这个值是为了优化查询,每次只取最新的最新一次评测的结果集
|
|
|
max_query_index = game.query_index - 1
|
|
|
# 判断实训是否全部通关
|
|
|
had_done = game.had_done
|
|
|
# 统计时间
|
|
|
record = EvaluateRecord.where(:game_id => game.id).first.try(:consume_time)
|
|
|
# 实训制作者当前拥有的金币
|
|
|
grade = myshixun.owner.try(:grade)
|
|
|
|
|
|
# power判断用户是否有权限查看隐藏测试集(TPM管理员;平台认证的老师;花费金币查看者) -1 表示不能解锁 0 表示需要付费解锁 1表示可以看
|
|
|
# myshixun_manager
|
|
|
myshixun_manager = shixun_manager(shixun, current_user) || (current_user.is_certification_teacher)
|
|
|
power = (myshixun_manager || game.test_sets_view ) ? 1 : (shixun.test_set_permission ? 0 : -1)
|
|
|
|
|
|
# 选择题和编程题公共部分
|
|
|
container = {:st => st, :discusses_count => discusses_count, :game_count => game_count, :myshixun => myshixun.try(:attributes),
|
|
|
:challenge => game_challenge.try(:attributes), :game => game.try(:attributes), :shixun => shixun.try(:attributes),
|
|
|
:record => record, :grade => grade, :prev_game => prev_game, :next_game => next_game, :username => username,
|
|
|
:image_url => image_url, :user_url => user_url, :praise_count => praise_count, :user_praise => user_praise, :time_limit => time_limit,
|
|
|
:tomcat_url => Redmine::Configuration['tomcat_php'], :is_teacher => is_teacher, :power => power, :myshixun_manager => myshixun_manager,
|
|
|
:local_exam => LocalSetting.first.try(:exam)}
|
|
|
|
|
|
# 区分选择题和编程题,st:0编程题;
|
|
|
if st == 0
|
|
|
gitUrl = gitlab_url myshixun
|
|
|
gitUrl = Base64.urlsafe_encode64(gitUrl)
|
|
|
shixun_tomcat = Redmine::Configuration['shixun_tomcat']
|
|
|
params = {:tpiID => "#{myshixun.id}", :tpiGitURL => "#{gitUrl}"}
|
|
|
|
|
|
# 判断tpm是否修改了
|
|
|
tpm_modified = repository_is_modified(myshixun, shixun.try(:gpid)) # 判断TPM和TPI的版本库是否被改了
|
|
|
tpm_cases_modified = (game_challenge.modify_time != game.modify_time ? true : false) # modify_time 决定TPM测试集是否有更新
|
|
|
# tpm_script_modified = (shixun.reset_time > myshixun.reset_time ? true : false) # 新结构脚本是实时传送的,所以不会影响
|
|
|
|
|
|
# 区分评测过未评测过,未评测过按需求取数据
|
|
|
sql =
|
|
|
if max_query_index > 0
|
|
|
"SELECT
|
|
|
b.code, b.actual_output, b.out_put, b.result, b.compile_success, a.is_public, a.input, a.output
|
|
|
FROM
|
|
|
(SELECT position, input, output, challenge_id, is_public FROM test_sets where challenge_id=#{game_challenge.id}) a
|
|
|
LEFT JOIN
|
|
|
(SELECT
|
|
|
result, test_set_position, g.challenge_id, o.actual_output, o.out_put, o.compile_success, o.code
|
|
|
FROM
|
|
|
outputs o left join games g on g.id=o.game_id
|
|
|
WHERE
|
|
|
game_id=#{game.id} and query_index = #{max_query_index}
|
|
|
) b
|
|
|
ON
|
|
|
b.challenge_id = a.challenge_id and b.test_set_position = a.position"
|
|
|
else
|
|
|
# 这个地方如果加字段的话一定注意test_set_static_data方法,因为这个地方的字段数在下面有判断
|
|
|
"SELECT t.is_public, t.input, t.output, t.position
|
|
|
FROM games g ,challenges c,test_sets t
|
|
|
WHERE g.id=#{game.id} and c.id= g.challenge_id and t.challenge_id = c.id"
|
|
|
end
|
|
|
qurey_test_sets = TestSet.find_by_sql(sql)
|
|
|
|
|
|
# 测试集统计及处理
|
|
|
unless qurey_test_sets.blank?
|
|
|
check_power = (power == 1 || game.test_sets_view)
|
|
|
test_sets = test_set_static_data(qurey_test_sets, check_power)
|
|
|
end
|
|
|
test_sets_count = qurey_test_sets.size
|
|
|
had_test = Output.where(:game_id => game.id, :query_index => max_query_index)
|
|
|
had_test_count = had_test.count
|
|
|
had_passed_testsests_error_count = had_test.blank? ? 0 : had_test.select{|had_test| had_test.result == false}.count
|
|
|
had_passed_testsests_error_count = had_test_count - had_passed_testsests_error_count
|
|
|
|
|
|
error_position = had_test.blank? ? nil : had_test.select{|had_test| had_test.result == false}.last
|
|
|
Rails.logger.info("latest output id is #{error_position.id unless error_position.blank?}")
|
|
|
latest_output = error_position.try(:out_put).gsub(/\n/, '<br/>').gsub(/\t/, " \; \; \; \; \; \; \; \;") unless error_position.try(:out_put).blank?
|
|
|
Rails.logger.warn(latest_output)
|
|
|
|
|
|
output_hash = {:test_sets => test_sets}.merge(:had_test_count => had_test_count, :test_sets_count => test_sets_count,
|
|
|
:had_passed_testsests_error_count => had_passed_testsests_error_count)
|
|
|
multi_webssh = shixun.webssh == 2 && shixun.multi_webssh
|
|
|
result = {:tpm_modified => tpm_modified, :tpm_cases_modified => tpm_cases_modified, :hide_code => shixun.hide_code, :forbid_copy => true,
|
|
|
:output_sets => output_hash, :latest_output => latest_output, :mirror_name => mirror_name, :multi_webssh => multi_webssh,
|
|
|
:has_answer => has_answer}.merge(container)
|
|
|
else # 选择题类型的
|
|
|
# 该方法多个地方调用,比如show、评测
|
|
|
# 最后一个字段true表示只显示数据,false表示可能有添加数据
|
|
|
result_container = choose_container(game_challenge, game, max_query_index, true)
|
|
|
return result_container.merge(container)
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# output为testset的实际输出
|
|
|
# out_put为顶端结果显示输出 power 查看测试集的权限
|
|
|
def test_set_static_data test_sets, power
|
|
|
test_result = []
|
|
|
unless test_sets.blank?
|
|
|
test_sets.each do |test_set|
|
|
|
# 第一次刷新或者没有评测的的时候test_set只需取四个字段,所以取了count
|
|
|
actual_output = test_set.attributes.count > 4 ? test_set.try(:actual_output) : nil
|
|
|
result = test_set.attributes.count > 4 ? test_set.try(:result) : nil
|
|
|
compile_success = test_set.attributes.count > 4 ? test_set.try(:compile_success) : nil
|
|
|
status = test_set.attributes.count > 4 ? test_set.try(:code) : nil
|
|
|
#actual_output = Base64.encode64(actual_output)
|
|
|
|
|
|
actual_output = (compile_success.to_s == "0" && status.to_s == "-1" ? "编译失败,请在测试结果中查看具体的错误信息" : actual_output)
|
|
|
|
|
|
public_result = {:is_public => (test_set.is_public ? 1 : 0), :result => result,
|
|
|
:actual_output => actual_output, :compile_success => compile_success}
|
|
|
# 测试集公开的话才返回input结果
|
|
|
(test_set.is_public || power) && public_result.merge!({:input => test_set.input, :output => test_set.output})
|
|
|
Rails.logger.info("#######################test_set: #{public_result}")
|
|
|
test_result << public_result.to_json
|
|
|
end
|
|
|
end
|
|
|
test_result = test_result.blank? ? test_result : test_result.join(",")
|
|
|
return test_result.gsub(/<\/script>/, '<//script>')
|
|
|
end
|
|
|
|
|
|
# 获取TPI关卡内容
|
|
|
def challenges params
|
|
|
result = []
|
|
|
myshixun = Myshixun.find(Game.where(:identifier => params[:identifier]).pluck(:myshixun_id).first)
|
|
|
shixun = Shixun.select([:id, :status]).find(myshixun.shixun_id)
|
|
|
games = myshixun.games.includes(:challenge).reorder("challenges.position")
|
|
|
games.each do |game|
|
|
|
game_status = game.try(:status)
|
|
|
challenge = game.challenge
|
|
|
if challenge.st == 0 # 实战类型
|
|
|
if game_status == 2 # 通关了则取实际得分,没通关则取总分
|
|
|
gold = (shixun.status <= 1) ? 0 : game.final_score.to_i
|
|
|
# 只要过关了,查看了答案经验值就是0,金币是负数
|
|
|
experience = (shixun.status <= 1 || game.final_score.to_i < 0) ? 0 : challenge.score.to_i
|
|
|
challenge_tags_count = (shixun.status <= 1) ? 0 : challenge.challenge_tags.count
|
|
|
else
|
|
|
gold = challenge.score.to_i
|
|
|
experience = gold
|
|
|
challenge_tags_count = challenge.challenge_tags.count
|
|
|
end
|
|
|
else
|
|
|
if game_status == 2
|
|
|
gold = (shixun.status <= 1) ? 0 : game.final_score.to_i
|
|
|
experience = (shixun.status <= 1 || game.final_score.to_i < 0) ? 0 : challenge.choose_score.to_i
|
|
|
challenge_tags_count = (shixun.status <= 1 || game.final_score.to_i <= 0) ? 0 : challenge.choose_tags_num
|
|
|
else
|
|
|
# 选择题只有在全对的时候才会获取final score总分,错任何一个题final_score就为0
|
|
|
gold = challenge.choose_score.to_i
|
|
|
experience = challenge.choose_score.to_i
|
|
|
challenge_tags_count = challenge.choose_tags_num
|
|
|
end
|
|
|
end
|
|
|
result << {:position => challenge.position, :subject => challenge.subject, :tag_count => challenge_tags_count, :gold => gold,
|
|
|
:experience => experience, :status => game.status, :star => game.star, :identifier => game.identifier}
|
|
|
end
|
|
|
result
|
|
|
end
|
|
|
|
|
|
# 评论打星星
|
|
|
def star params, current_user
|
|
|
game = Game.find_by_identifier(params[:identifier])
|
|
|
shixun = Shixun.select([:id, :status]).find(params[:shixun_id])
|
|
|
# 如果已经评星,则不能重复评
|
|
|
grade = Grade.where(:user_id => current_user.id, :container_id => game.id, :container_type => 'Star').first
|
|
|
if grade.present?
|
|
|
return{:reward_code => -1}
|
|
|
else
|
|
|
game.update_column(:star, params[:star].to_i)
|
|
|
# 随机生成10-100金币
|
|
|
code = rand(10..100)
|
|
|
# 积分消耗情况记录;加积分之针对已发布的实训
|
|
|
if shixun.status >= 2
|
|
|
reward_grade(current_user, game.id, 'Star', code)
|
|
|
return{:reward_code => code}
|
|
|
else
|
|
|
return{:reward_code => 0}
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# TPI版本库文件目录结构
|
|
|
def entries params
|
|
|
g = Gitlab.client
|
|
|
gpid = params[:gpid]
|
|
|
path = params[:path].try(:strip)
|
|
|
rev = params[:rev] ? params[:rev] : "master"
|
|
|
result = []
|
|
|
trees = g.trees(gpid, path: path, rev: rev)
|
|
|
trees.each do |tree|
|
|
|
# 去掉第一行的"/"
|
|
|
new_path = File.join(path,tree.name).reverse.chomp("/").reverse
|
|
|
result << {:name => tree.name, :path => new_path, :kind => tree.type == 'tree' ? 'dir' : 'file'}
|
|
|
end
|
|
|
|
|
|
return result
|
|
|
end
|
|
|
|
|
|
def answer params, current_user
|
|
|
game_base params[:identifier]
|
|
|
challenge = Challenge.select([:answer, :id, :score, :st]).find(@game.challenge_id)
|
|
|
is_teacher = (current_user.user_extensions.identity == 0)
|
|
|
is_certification_teacher = current_user.is_certification_teacher
|
|
|
# 这几种情况可以直接查看答案的:实训未发布;当前用户为实训管理员;已经查看过答案;平台认证的老师;
|
|
|
if (@shixun.status < 2 || shixun_manager(@shixun, current_user) || @game.answer_open || is_certification_teacher)
|
|
|
Rails.logger.info("answer#{params[:identifier]}#### user is #{current_user.id}, status is #{@shixun.status}, manager is #{shixun_manager(@shixun, current_user)}, open #{@game.answer_open}, cer #{is_certification_teacher}")
|
|
|
if challenge.st == 0
|
|
|
result = challenge.try(:answer)
|
|
|
else
|
|
|
result = []
|
|
|
challenge.challenge_chooses.each do |choose|
|
|
|
result << {:position => choose.position, :answer => (choose.answer.blank? ? choose.standard_answer : choose.answer)}
|
|
|
end
|
|
|
end
|
|
|
return {:view_answer => true, :answer => result}
|
|
|
else
|
|
|
return {:is_teacher => is_teacher, :view_answer => false}
|
|
|
# view_answer = false
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 查看答案需要扣取金币
|
|
|
# 必须保证用户的金币数大于关卡的金币数
|
|
|
def answer_grade params, current_user
|
|
|
game_base params[:identifier]
|
|
|
challenge = Challenge.select([:answer, :id, :score, :st]).find(@game.challenge_id)
|
|
|
challenge_score = challenge.try(:score)
|
|
|
final_score = @game.final_score
|
|
|
if current_user.grade.to_i - challenge_score > 0
|
|
|
if !@game.answer_open # 如果这是第一次查看答案
|
|
|
if challenge.st == 0
|
|
|
final_score = final_score - challenge_score
|
|
|
# 积分消耗情况记录
|
|
|
reward_grade(current_user, @game.id, 'Answer', -challenge_score)
|
|
|
else
|
|
|
final_score = final_score - challenge.choose_score.to_i
|
|
|
# 之所以不用final_score是因为过关后查看答案的final_score为0,但是记录需要记录扣除的分数
|
|
|
reward_grade(current_user, @game.id, 'Answer', -challenge.choose_score.to_i)
|
|
|
end
|
|
|
@game.update_attributes!(:answer_open => true, :final_score => final_score)
|
|
|
end
|
|
|
else
|
|
|
# 金币不足
|
|
|
raise("0")
|
|
|
end
|
|
|
if challenge.st == 0
|
|
|
answer = challenge.try(:answer)
|
|
|
else
|
|
|
answer = []
|
|
|
challenge.challenge_chooses.each do |choose|
|
|
|
answer << {:position => choose.position, :answer => (choose.answer.blank? ? choose.standard_answer : choose.answer)}
|
|
|
end
|
|
|
end
|
|
|
|
|
|
return {:answer => answer, :final_score => final_score, :grade => @myshixun.owner.try(:grade)}
|
|
|
end
|
|
|
|
|
|
# 查看隐藏测试集
|
|
|
def check_test_sets params, current_user
|
|
|
game = Game.select([:id, :challenge_id, :test_sets_view]).find_by_identifier(params[:identifier])
|
|
|
if game.test_sets_view == true
|
|
|
return {:status => -1, :message => "已解锁,请误重复操作!"}
|
|
|
end
|
|
|
challenge = Challenge.select([:id, :score]).find(game.challenge_id)
|
|
|
user_grade = current_user.grade
|
|
|
minus_grade = challenge.score * 5
|
|
|
max_query_index = game.query_index - 1
|
|
|
# 区分评测过未评测过,未评测过按需求取数据
|
|
|
if max_query_index > 0
|
|
|
qurey_test_sets = TestSet.find_by_sql("SELECT o.code, o.actual_output, o.out_put, o.result, o.compile_success, o.test_set_position, o.query_index,t.is_public,t.input, t.output, g.id as game_id, c.id as challenge_id FROM outputs o,games g ,challenges c,test_sets t where
|
|
|
g.id=#{game.id} and o.query_index=#{max_query_index} and g.id = o.game_id and c.id= g.challenge_id and t.challenge_id = c.id and t.position =o.test_set_position order by o.query_index
|
|
|
")
|
|
|
else
|
|
|
qurey_test_sets = TestSet.find_by_sql("SELECT t.is_public,t.input, t.output,t.position FROM games g ,challenges c,test_sets t where
|
|
|
g.id=#{game.id} and c.id= g.challenge_id and t.challenge_id = c.id
|
|
|
")
|
|
|
end
|
|
|
test_sets = test_set_static_data(qurey_test_sets, true)
|
|
|
if user_grade >= minus_grade
|
|
|
current_user.update_attribute(:grade, user_grade - minus_grade)
|
|
|
game.update_attribute(:test_sets_view, true)
|
|
|
# 扣分记录
|
|
|
gardes = Grade.create(:user_id => current_user.id, :container_id => game.id, :score => -minus_grade, :container_type => "testSet")
|
|
|
gardes.attributes.merge!({test_sets: test_sets})
|
|
|
else
|
|
|
return {:status => -1, :message => "本操作需要扣除#{ minus_grade }金币,您的金币不够了"}
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 文件更新;数据评测记录
|
|
|
# 生成重新评测认证码
|
|
|
# content_modified:0 表示文件没有更新;content_modified:1 表示文件有更新
|
|
|
def file_update params, current_user
|
|
|
game_base params[:identifier]
|
|
|
path = params[:path].strip unless params[:path].blank?
|
|
|
rev = params[:rev] ? params[:rev] : "master"
|
|
|
content_modified = 0
|
|
|
ActiveRecord::Base.transaction do
|
|
|
# params[:evaluate] 实训评测时更新必须给的参数,需要依据该参数做性能统计,其它类型的更新可以跳过
|
|
|
# 自动保存的时候evaluate为0;点评测的时候为1
|
|
|
if params[:evaluate] == 1
|
|
|
record = EvaluateRecord.create!(:user_id => current_user.id, :shixun_id => @myshixun.shixun.id, :game_id => @game.id)
|
|
|
Rails.logger.warn("##game is is #{@game.id}, record id is #{record.id}, time is**** #{Time.now.strftime("%Y-%m-%d %H:%M:%S.%L")}")
|
|
|
# @myshixun.student_works.update_all(:update_time => Time.now) if !@myshixun.student_works.blank?
|
|
|
student_work_time = format("%.3f", (Time.now.to_f - record.created_at.to_f)).to_f
|
|
|
record.update_attributes!(:student_work => student_work_time)
|
|
|
end
|
|
|
|
|
|
@g = Gitlab.client
|
|
|
last_content = @g.files(@myshixun.gpid, path, "master").try(:content)
|
|
|
|
|
|
last_content = tran_base64_decode64(last_content)
|
|
|
content = if @myshixun.mirror_name.select{|a| a.include?("MachineLearning") || a.include?("Python")}.present? && params[:content].present?
|
|
|
params[:content].gsub(/\t/, ' ')
|
|
|
else
|
|
|
params[:content]
|
|
|
end
|
|
|
if content != last_content
|
|
|
content_modified = 1
|
|
|
code_file = @g.edit_file(@myshixun.gpid, current_user.login, :content => content, :file_path => path, :branch_name => "master", :commit_message => "auto commit")
|
|
|
end
|
|
|
# REDO:是否有读写分离的问题
|
|
|
if record.present?
|
|
|
consume_time = format("%.3f", (Time.now.to_f - record.created_at.to_f)).to_f
|
|
|
record.update_attributes!(:file_update => consume_time)
|
|
|
end
|
|
|
|
|
|
# status为2说明是重新评测
|
|
|
if @game.status == 2
|
|
|
code = CODES.sample(8).join
|
|
|
resubmit = "#{code}_#{@myshixun.id}"
|
|
|
end
|
|
|
|
|
|
if content != last_content && code_file.blank?
|
|
|
raise("update file failed")
|
|
|
else
|
|
|
return {:success => "success", :resubmit => resubmit ,:content_modified => content_modified}
|
|
|
end
|
|
|
end
|
|
|
rescue Exception => e
|
|
|
Rails.logger.error("file update failed #{e.message}")
|
|
|
return {:status => -1, :success => "fail", :message => "实训云平台繁忙(繁忙等级:81),请稍后刷新并重试", :position => @game.challenge.position, :had_done => @game.had_done}
|
|
|
end
|
|
|
|
|
|
# 恢复初始代码
|
|
|
# 注意path为当前打开文件的path
|
|
|
def reset_original_code params, current_user
|
|
|
g = Gitlab.client
|
|
|
path = params[:path]
|
|
|
begin
|
|
|
file_base params[:identifier]
|
|
|
shixun = Shixun.select(:gpid).find(@myshixun.shixun_id)
|
|
|
content = g.files(shixun.gpid, path, "master").try(:content)
|
|
|
content = tran_base64_decode64(content)
|
|
|
g.edit_file(@myshixun.gpid, current_user.login, :content => content, :file_path => path, :branch_name => "master",
|
|
|
:commit_message => "reset_original_code")
|
|
|
if content.nil?
|
|
|
raise("初始代码为空,代码重置失败")
|
|
|
end
|
|
|
return content
|
|
|
rescue Exception => e
|
|
|
raise("#{e.message}")
|
|
|
end
|
|
|
end
|
|
|
|
|
|
def reset_passed_code params, current_user
|
|
|
g = Gitlab.client
|
|
|
path = params[:path]
|
|
|
game= Game.select([:id, :myshixun_id]).where(:identifier => params[:identifier]).first
|
|
|
myshixun = Myshixun.find(game.myshixun_id)
|
|
|
game_code = GameCode.where(:game_id => game.try(:id), :path => path).first
|
|
|
if game_code.present?
|
|
|
content = game_code.try(:new_code)
|
|
|
content = if myshixun.mirror_name.select{|a| a.include?("MachineLearning") || a.include?("Python")}.present? && content.present?
|
|
|
content.gsub(/\t/, ' ')
|
|
|
else
|
|
|
content
|
|
|
end
|
|
|
g.edit_file(myshixun.try(:gpid), current_user.login, :content => content, :file_path => path, :branch_name => "master", :commit_message => "game passed reset")
|
|
|
return content
|
|
|
else
|
|
|
return {:status => -1, :message => "重置失败,代码为空"}
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 获取版本库文件内容
|
|
|
# 通过版本库的默认分支可以判断版本库是否正常
|
|
|
# 注:如果本身path传错,内容肯定也为空;fork成功后,可能短时间内也获取不到版本库内容
|
|
|
# status =0 表示获取默认打开的内容,1表示右侧目录结构中获取的内容
|
|
|
def rep_content params, current_user
|
|
|
file_base params[:identifier]
|
|
|
# current_user: fork不应该是用当前用户去fork
|
|
|
current_user = @myshixun.owner
|
|
|
g = Gitlab.client
|
|
|
gpid = @myshixun.try(:gpid)
|
|
|
path = @game.challenge.try(:path).split(";")[0].strip()
|
|
|
path = params[:path] ? params[:path] : path
|
|
|
shixun_gpid = params[:shixun_gpid]
|
|
|
status = params[:status]
|
|
|
path = path.try(:strip)
|
|
|
rev = params[:rev] ? params[:rev] : "master"
|
|
|
Rails.logger.warn("path is #{path}")
|
|
|
content = g.files(gpid, path, rev).try(:content)
|
|
|
|
|
|
if content.blank?
|
|
|
# 监测版本库HEAD是否存在,不存在则取最新的HEAD
|
|
|
gitUrl = gitlab_url @myshixun
|
|
|
gitUrl = Base64.urlsafe_encode64(gitUrl)
|
|
|
shixun_tomcat = Redmine::Configuration['shixun_tomcat']
|
|
|
rep_params = {:tpiID => "#{@myshixun.id}", :tpiGitURL => "#{gitUrl}"}
|
|
|
# 监测版本库HEAD是否存在,不存在则取最新的HEAD
|
|
|
uri = "#{shixun_tomcat}/bridge/game/check"
|
|
|
res = uri_exec uri, rep_params
|
|
|
# res值:0 表示正常;-1表示有错误;-2表示代码版本库没了
|
|
|
if (status == 0 && res && res['code'] == -2 && params[:retry].to_i == 1)
|
|
|
# 没有版本库则删除现有的版本库关联,重新fork一个版本库
|
|
|
g = Gitlab.client
|
|
|
if current_user.gid.nil?
|
|
|
s = Trustie::Gitlab::Sync.new
|
|
|
s.sync_user(current_user)
|
|
|
end
|
|
|
gpid = sync_gitlab_rep @myshixun, shixun_gpid, current_user.try(:gid)
|
|
|
content = g.files(gpid, path, rev).content
|
|
|
else
|
|
|
if (res && res['code'] != 0)
|
|
|
Rails.logger.error("get rep_content failed: bridge return !0")
|
|
|
# raise(status)
|
|
|
return nil
|
|
|
end
|
|
|
end
|
|
|
|
|
|
|
|
|
# gitlab缺陷:forked完成,短暂时间内取不了内容的,所以做一个小轮询,间隔0.2秒
|
|
|
# 超过3秒则失败,需通过页面刷新
|
|
|
# for i in 0..15 do
|
|
|
# sleep(0.2)
|
|
|
# content = g.files(gpid, path, rev).content
|
|
|
# break if content.present?
|
|
|
# end
|
|
|
end
|
|
|
|
|
|
content = tran_base64_decode64(content) if content.present?
|
|
|
return content
|
|
|
end
|
|
|
|
|
|
|
|
|
# 编程题评测
|
|
|
def game_build params
|
|
|
# 三类实训只取基础部分数据
|
|
|
game = Game.select([:myshixun_id, :status, :challenge_id, :id, :evaluate_count]).find_by_identifier(params[:identifier])
|
|
|
myshixun = Myshixun.select([:updated_at, :gpid, :id, :shixun_id]).find(game.myshixun_id)
|
|
|
shixun = Shixun.select([:id, :evaluate_script, :webssh, :exec_time, :sigle_training, :identifier, :status]).find(myshixun.shixun_id)
|
|
|
game_challenge = Challenge.select([:id, :position, :picture_path]).find(game.challenge_id)
|
|
|
|
|
|
# 更新评测次数
|
|
|
game.update_column(:evaluate_count, (game.evaluate_count.to_i + 1))
|
|
|
# 清空代码评测信息
|
|
|
# msg = game.run_code_message
|
|
|
# msg.update_attributes(:status => 0, :message => nil) if msg.present?
|
|
|
# 更新时间是为了TPM端显示的更新,退出实训及访问实训的时候会更新
|
|
|
myshixun.update_column(:updated_at, Time.now)
|
|
|
|
|
|
gitUrl = gitlab_url myshixun
|
|
|
gitUrl = Base64.urlsafe_encode64(gitUrl)
|
|
|
taskId = game.id
|
|
|
shixun_tomcat = Redmine::Configuration['shixun_tomcat']
|
|
|
step = game_challenge.try(:position)
|
|
|
mirror_repository_limit = shixun.mirror_repositories.where(:main_type => 1).select(:resource_limit).try(:first).try(:resource_limit)
|
|
|
# mirror表中很很大的脚本字段,所以单独查询一个字段效果更好
|
|
|
resource_limit = "echo 'ulimit -f #{mirror_repository_limit}' >> /root/.bashrc ; source /root/.bashrc\n"
|
|
|
tpmScript = shixun.evaluate_script.nil? ? "" : Base64.urlsafe_encode64((resource_limit + shixun.evaluate_script).gsub("\r\n", "\n"))
|
|
|
|
|
|
# status为2已经通过关,是重新评测
|
|
|
if game.status == 2
|
|
|
resubmit = params[:resubmit]
|
|
|
else
|
|
|
# 重新评测不影响已通关的实训状态;first为第一次评测,通过前端JS轮询获取
|
|
|
game.update_attributes!(:status => 1) if params[:first].to_i == 1
|
|
|
end
|
|
|
|
|
|
testSet = []
|
|
|
game_challenge.test_sets.each do |test_set|
|
|
|
input = test_set.input.nil? ? "" : test_set.input.gsub("\r\n", "\n")
|
|
|
output = test_set.output.nil? ? "" : test_set.output.gsub("\r\n", "\n")
|
|
|
test_cases = {:input => input, :output => output}
|
|
|
testSet << test_cases
|
|
|
end
|
|
|
|
|
|
testCases = Base64.urlsafe_encode64(testSet.to_json) unless testSet.blank?
|
|
|
|
|
|
|
|
|
# 注意:这个地方的参数写的时候不能换行
|
|
|
content_modified = params[:content_modified] # 决定文件内容是否有修改,有修改如果中间成pull没有更新,则轮询等待更新
|
|
|
params = {:tpiID => "#{myshixun.id}", :tpiGitURL => "#{gitUrl}", :buildID => "#{taskId}",:instanceChallenge => "#{step}",
|
|
|
:testCases => "#{testCases}", :resubmit => "#{resubmit}", :times => params[:first].to_i, :podType => shixun.webssh,
|
|
|
:containers => "#{Base64.urlsafe_encode64(container_limit(shixun.mirror_repositories))}", :tpmScript => "#{tpmScript}",
|
|
|
:timeLimit => "#{shixun.exec_time}", :content_modified => content_modified, :persistenceName => shixun.identifier,
|
|
|
:isPublished => (shixun.status < 2 ? 0 : 1)}
|
|
|
|
|
|
# 评测有文件输出的需要特殊传字段 path:表示文件存储的位置
|
|
|
params['file'] = Base64.urlsafe_encode64({:path => "#{game_challenge.picture_path}"}.to_json) if game_challenge.picture_path.present?
|
|
|
|
|
|
# needPortMapping: web类型需要pod端口映射
|
|
|
params[:needPortMapping] = 8080 if myshixun.mirror_name.include?("Web")
|
|
|
|
|
|
# 中间层交互
|
|
|
if shixun.sigle_training
|
|
|
uri = "#{shixun_tomcat}/bridge/game/persistence/gameEvaluate"
|
|
|
else
|
|
|
uri = "#{shixun_tomcat}/bridge/game/gameEvaluate"
|
|
|
end
|
|
|
|
|
|
res = interface_post uri, params, 502
|
|
|
# 单评测类型(比较快的类型,实时返回结果,即不用中间层再回调trainint_task_status)
|
|
|
if res['syncResult'].present?
|
|
|
sigle_trainint_data res, myshixun, game
|
|
|
end
|
|
|
# ----单测模式end
|
|
|
|
|
|
return {:result => "success", :resubmit => resubmit, :ableToCreate => res['ableToCreate'], :waitNum => res['waitNum'],
|
|
|
:waitingTime => res['waitingTime'], :position => game_challenge.position, :port => res['port'], :had_done => game.had_done }
|
|
|
rescue Exception => e
|
|
|
Rails.logger.error("评测出错,详情:" + e.message)
|
|
|
return {:result => 'fail', :contents =>"实训云平台繁忙(繁忙等级:502),请稍后刷新并重试", :position => game_challenge.position, :had_done => game.had_done}
|
|
|
end
|
|
|
|
|
|
# 单评测类型(比较快的类型,实时返回结果,即不用中间层再回调trainint_task_status)
|
|
|
def sigle_trainint_data res, myshixun, game
|
|
|
begin
|
|
|
t1 = Time.now
|
|
|
jsonTestDetails = JSON.parse(res['syncResult']['jsonTestDetails'])
|
|
|
Rails.logger.info("res is #{res} ### jsonTestDetails is #{jsonTestDetails}")
|
|
|
timeCost = JSON.parse(res['syncResult']['timeCost'])
|
|
|
brige_end_time = Time.parse(timeCost['evaluateEnd']) if timeCost['evaluateEnd'].present?
|
|
|
return_back_time = format("%.3f", ( t1.to_f - brige_end_time.to_f)).to_f
|
|
|
status = jsonTestDetails['status']
|
|
|
game_id = jsonTestDetails['buildID']
|
|
|
Rails.logger.info("sigle training_task_status start#1**#{game_id}**** #{Time.now.strftime("%Y-%m-%d %H:%M:%S.%L")}")
|
|
|
resubmit = jsonTestDetails['resubmit']
|
|
|
outPut = tran_base64_decode64(jsonTestDetails['outPut'])
|
|
|
jenkins_testsets = jsonTestDetails['msg']
|
|
|
compile_success = jsonTestDetails['compileSuccess']
|
|
|
Rails.logger.info(outPut)
|
|
|
challenge = game.challenge
|
|
|
if challenge.picture_path.present?
|
|
|
pics = res[:tpiRepoPath]
|
|
|
game.update_column(:picture_path, pics)
|
|
|
end
|
|
|
Rails.logger.info("training_task_status start#2**#{game_id}**** #{Time.now.strftime("%Y-%m-%d %H:%M:%S.%L")}")
|
|
|
unless jenkins_testsets.blank?
|
|
|
jenkins_testsets.each do |j_test_set|
|
|
|
Rails.logger.info("j_test_set: ############## #{j_test_set}")
|
|
|
actual_output = tran_base64_decode64(j_test_set['output'])
|
|
|
Rails.logger.info "actual_output:################################################# #{actual_output}"
|
|
|
Output.create!(:code => status, :game_id => game_id, :out_put => outPut, :test_set_position => j_test_set['caseId'],
|
|
|
:actual_output => actual_output, :result => j_test_set['passed'].to_i, :query_index => game.query_index,
|
|
|
:compile_success => compile_success.to_i)
|
|
|
end
|
|
|
end
|
|
|
Rails.logger.info("#############status: #{status}")
|
|
|
record = EvaluateRecord.where(:game_id => game_id).first
|
|
|
Rails.logger.info("training_task_status start#3**#{game_id}**** #{Time.now.strftime("%Y-%m-%d %H:%M:%S.%L")}")
|
|
|
|
|
|
# status:0表示评测成功
|
|
|
if status == "0"
|
|
|
if !resubmit.blank?
|
|
|
game.update_attributes!(:retry_status => 2, :resubmit_identifier => resubmit)
|
|
|
#if game.had_done == 1
|
|
|
challenge.path.split(";").each do |path|
|
|
|
game_passed_code(game.id, path.try(:strip), myshixun.try(:gpid), 1)
|
|
|
end
|
|
|
#end
|
|
|
else
|
|
|
game.update_attributes!(:status => 2, :end_time => Time.now, :accuracy => format("%.4f", 1.0 / game.query_index))
|
|
|
myshixun.update_attributes!(:status => 1) if game.had_done == 1
|
|
|
challenge.path.split(";").each do |path|
|
|
|
game_passed_code(game.id, path, myshixun.try(:gpid), 1)
|
|
|
end
|
|
|
if !game.answer_open && (challenge.shixun.try(:status) > 1) # 如果没有查看答案,则获得该关卡得分
|
|
|
reward_grade(game.user, game.id, 'Game', challenge.score)
|
|
|
reward_experience(game.user, game.id, 'Game', challenge.score)
|
|
|
game.update_attributes!(:final_score => challenge.score)
|
|
|
end
|
|
|
|
|
|
# 更新实训关联的作品分数
|
|
|
update_myshixun_work_score myshixun
|
|
|
end
|
|
|
# 如果过关了,下一关的状态是3(为开启),则需要把状态改成1(已开启)
|
|
|
# next_game = game.next_game
|
|
|
next_game = Game.next_game(myshixun.shixun_id, game.myshixun_id, challenge.position)
|
|
|
next_game.update_column(:status, 0) if next_game.present? && next_game.status == 3
|
|
|
else # status == "-1" 表示返回结果错误
|
|
|
if !resubmit.blank?
|
|
|
game.update_attributes!(:retry_status => 1, :resubmit_identifier => resubmit)
|
|
|
else
|
|
|
game.update_attributes!(:status => 0)
|
|
|
end
|
|
|
end
|
|
|
test_cases_time = format("%.3f", (Time.now.to_f - t1.to_f)).to_f
|
|
|
if record.present?
|
|
|
consume_time = format("%.3f", (Time.now - record.created_at)).to_f
|
|
|
record.update_attributes!(:consume_time => consume_time, :git_pull => timeCost['pull'] ,
|
|
|
:create_pod => timeCost['createPod'], :pod_execute => timeCost['execute'], :test_cases => test_cases_time,
|
|
|
:brige => timeCost['evaluateAllTime'], :return_back => return_back_time)
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 选择题评测
|
|
|
def choose_build params
|
|
|
# 三类实训只取基础部分数据
|
|
|
game = Game.select([:myshixun_id, :status, :challenge_id, :id, :answer_open, :final_score, :identifier,
|
|
|
:evaluate_count]).find_by_identifier(params[:identifier])
|
|
|
# 更新评测次数
|
|
|
game.update_column(:evaluate_count, (game.evaluate_count.to_i + 1))
|
|
|
|
|
|
# 选择题如果通关了,则不让再评测
|
|
|
if game.status == 2
|
|
|
return {:message => "您已通过该关卡"}
|
|
|
end
|
|
|
|
|
|
myshixun = Myshixun.select([:updated_at, :gpid, :id, :shixun_id, :status, :user_id]).find(game.myshixun_id)
|
|
|
shixun = Shixun.select([:id, :evaluate_script, :webssh, :status]).find(myshixun.shixun_id)
|
|
|
game_challenge = Challenge.select([:id, :position]).find(game.challenge_id)
|
|
|
user_answer = params[:answer]
|
|
|
challenge_chooses_count = user_answer.length
|
|
|
|
|
|
choose_correct_num = 0
|
|
|
score = 0
|
|
|
total_score = 0
|
|
|
had_passed = true
|
|
|
test_sets = []
|
|
|
str = ""
|
|
|
game_challenge.challenge_chooses.includes(:challenge_tags).each_with_index do |choose, index|
|
|
|
user_answer_tran = user_answer[index].size > 1 ? user_answer[index].split("").sort.join("") : user_answer[index]
|
|
|
standard_answer_tran = choose.standard_answer.size > 1 ? choose.standard_answer.split("").sort.join("") : choose.standard_answer
|
|
|
correct = (user_answer_tran == standard_answer_tran) ? true : false
|
|
|
if str.present?
|
|
|
str += ","
|
|
|
end
|
|
|
str += "('#{game.id}', '#{choose.position}', '#{user_answer_tran}', '#{correct ? 1 : 0}', '#{game.query_index}', '#{Time.now.strftime('%Y-%m-%d %H:%M:%S')}', '#{Time.now.strftime('%Y-%m-%d %H:%M:%S')}')"
|
|
|
# Output.create(:game_id => @game.id, :test_set_position => choose.position, :actual_output => params[:answer][index], :result => correct, :query_index => @game.query_index)
|
|
|
# 只要有一题错误就不能通关
|
|
|
had_passed = false if !correct
|
|
|
choose_correct_num += 1 if correct
|
|
|
|
|
|
# 全部通关的时候,需要对所得的总分记录
|
|
|
# 总分的记录应该是根据每一题累加,如果该题正确,则加分
|
|
|
score += choose.score if correct
|
|
|
|
|
|
standard_answer = correct ? standard_answer_tran : -1
|
|
|
sin_test_set = {:result => correct, :actual_output => user_answer_tran, :standard_answer => standard_answer, :position => choose.position}
|
|
|
test_sets << sin_test_set
|
|
|
end
|
|
|
|
|
|
# 批量插入评测结果
|
|
|
Rails.logger.warn("sql str is #{str}")
|
|
|
sql = "INSERT INTO outputs (game_id, test_set_position, actual_output, result, query_index, created_at, updated_at) VALUES" + str
|
|
|
ActiveRecord::Base.connection.execute sql
|
|
|
|
|
|
|
|
|
# 没通关或者查看了答案通关的时候经验为0
|
|
|
# 通关但是查看了答案,评测的时候金币显示0(避免用户以为重复扣除),但是在关卡列表中金币显示负数
|
|
|
experience = 0
|
|
|
final_score = 0
|
|
|
# 如果本次答题全部正确,并且之前没有通关,则进行相应的奖励,每关只要有错题,则不进行任何奖励
|
|
|
# 注:扣除分数是在查看答案的时候扣除的
|
|
|
if had_passed && !game.had_passed?
|
|
|
game.update_attributes(:status => 2, :end_time => Time.now)
|
|
|
# TPM实训已发布并且没有查看答案
|
|
|
if shixun.is_published? && !game.answer_open
|
|
|
# 查看答案的时候处理final_scor和扣分记录
|
|
|
experience = score
|
|
|
reward_grade(myshixun.owner, game.id, 'Game', score)
|
|
|
game.update_attribute(:final_score, score)
|
|
|
final_score = score
|
|
|
reward_experience(myshixun.owner, game.id, 'Game', score)
|
|
|
end
|
|
|
end
|
|
|
|
|
|
had_done = game.had_done
|
|
|
myshixun.update_attribute(:status, 1) if had_done == 1
|
|
|
|
|
|
grade = myshixun.owner.try(:grade)
|
|
|
|
|
|
# 更新实训关联的作品分数
|
|
|
update_myshixun_work_score myshixun
|
|
|
|
|
|
# 高性能取上一关、下一关
|
|
|
prev_game = Game.prev_identifier(shixun.id, game.myshixun_id, game_challenge.position)
|
|
|
next_game = Game.next_game(shixun.id, game.myshixun_id, game_challenge.position)
|
|
|
# 如果过关了,下一关的状态是3(为开启),则需要把状态改成1(已开启)
|
|
|
next_game.update_column(:status, 0) if next_game.present? && next_game.status == 3
|
|
|
|
|
|
return {:grade => grade, :gold => final_score, :experience => experience, :had_submmit => true,
|
|
|
:challenge_chooses_count => challenge_chooses_count, :choose_correct_num => choose_correct_num,
|
|
|
:test_sets => test_sets, :prev_game => prev_game, :next_game => next_game.try(:identifier)}
|
|
|
end
|
|
|
|
|
|
# 实训选择题需要局部刷新或者显示的部分
|
|
|
def choose_container game_challenge, game, max_query_index, just_show
|
|
|
# category 1: 单选题,其它的多选题(目前只有两种)
|
|
|
challenge_chooses = game_challenge.challenge_chooses.includes(:challenge_questions)
|
|
|
test_sets = []
|
|
|
choose_test_cases = []
|
|
|
chooses = []
|
|
|
|
|
|
# 选择题测试集统计
|
|
|
challenge_chooses_count = challenge_chooses.count
|
|
|
choose_correct_num = game.choose_correct_num
|
|
|
game_outputs = game.outputs.where(:query_index => max_query_index)
|
|
|
|
|
|
# 判断用户是否有提交
|
|
|
had_submmit = game_outputs.blank? ? false : true
|
|
|
|
|
|
# 判断选择题是否写了标准答案
|
|
|
has_answer = []
|
|
|
challenge_chooses.each do |choose|
|
|
|
challenge_question = []
|
|
|
output = game_outputs.select{|game_output| game_output.test_set_position == choose.position}[0] unless game_outputs.blank?
|
|
|
|
|
|
category = choose.category
|
|
|
subject = choose.subject
|
|
|
choose.challenge_questions.each do |question|
|
|
|
position = question.position
|
|
|
option_name = question.option_name
|
|
|
challenge_question << {:positon => position, :option_name => option_name}
|
|
|
end
|
|
|
# actual_output为空表示暂时没有评测答题,不云讯查看
|
|
|
actual_output = output.try(:actual_output).try(:strip)
|
|
|
has_answer << choose.answer if choose.answer.present?
|
|
|
# 标准答案处理,错误的不让用户查看,用-1替代
|
|
|
standard_answer = (actual_output.blank? || !output.try(:result)) ? -1 : choose.standard_answer
|
|
|
result = output.try(:result)
|
|
|
sin_test_set = {:result => result, :actual_output => actual_output, :standard_answer => standard_answer, :position => choose.position}
|
|
|
|
|
|
sin_choose = {:category => category, :subject => subject, :challenge_question => challenge_question}
|
|
|
chooses << sin_choose
|
|
|
test_sets << sin_test_set
|
|
|
end
|
|
|
has_answer = has_answer.present?
|
|
|
choose_test_cases = {:had_submmit => had_submmit, :challenge_chooses_count => challenge_chooses_count, :choose_correct_num => choose_correct_num, :test_sets => test_sets}
|
|
|
return {:choose => chooses, :choose_test_cases => choose_test_cases, :has_answer => has_answer}
|
|
|
end
|
|
|
|
|
|
# 轮询获取状态
|
|
|
# resubmit是在file_update中生成的,从game_build中传入的
|
|
|
def game_status params, current_user
|
|
|
game = Game.find_by_identifier(params[:identifier])
|
|
|
resubmit_identifier = game.resubmit_identifier
|
|
|
# 如果没有超时并且正在评测中
|
|
|
# 判断评测中的状态有两种:1、如果之前没有通关的,只需判断status为1即可;如果通过关,则判断game的resubmit_identifier是否更新
|
|
|
if (params[:time_out] == "false") && ((params[:resubmit].blank? && game.status==1) || (params[:resubmit].present? && (params[:resubmit] != resubmit_identifier)))
|
|
|
# 代码评测的信息
|
|
|
#running_code_status = game.run_code_message.try(:status)
|
|
|
#running_code_message = game.run_code_message.try(:message)
|
|
|
return #{:running_code_status => running_code_status, :running_code_message => running_code_message}
|
|
|
end
|
|
|
|
|
|
Rails.logger.info("##### resubmit_identifier is #{resubmit_identifier}")
|
|
|
port = params[:port]
|
|
|
score = 0
|
|
|
experience = 0
|
|
|
game_status = game.status
|
|
|
had_done = game.had_done
|
|
|
game_challenge = Challenge.select([:id, :score, :position, :shixun_id, :web_route]).find(game.challenge_id)
|
|
|
shixun = Shixun.select([:id, :status, :user_id, :test_set_permission]).find(game_challenge.shixun_id)
|
|
|
|
|
|
if params[:resubmit].blank? # 非重新评测
|
|
|
if game_status == 2 # 通关
|
|
|
if shixun.status > 1
|
|
|
score = game.final_score # 查看答案的时候有对最终获得金币进行处理
|
|
|
experience = game.answer_open ? 0 : game_challenge.score.to_i
|
|
|
else
|
|
|
score = 0
|
|
|
experience = 0
|
|
|
end
|
|
|
end
|
|
|
else # 重新评测
|
|
|
# 如果满足前面的条件,进入此处只可能是结果已返回并存入了数据库
|
|
|
if params[:resubmit] == resubmit_identifier # 本次重新评测结果已经返回并存入数据库
|
|
|
game_status = (game.retry_status == 2 ? 2 : 0) # retry_status是判断重新评测的通关情况。2表示通关
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 实训的最大评测次数,这个值是为了优化查询,每次只取最新的最新一次评测的结果集
|
|
|
max_query_index = game.query_index - 1
|
|
|
# 区分评测过未评测过,未评测过按需求取数据
|
|
|
if max_query_index > 0
|
|
|
qurey_test_sets = TestSet.find_by_sql("SELECT o.code, o.actual_output,o.result, o.compile_success, o.test_set_position, o.query_index,t.is_public,t.input, t.output, g.id as game_id, c.id as challenge_id FROM outputs o,games g ,challenges c,test_sets t where
|
|
|
g.id=#{game.id} and o.query_index=#{max_query_index} and g.id = o.game_id and c.id= g.challenge_id and t.challenge_id = c.id and t.position =o.test_set_position order by o.query_index
|
|
|
")
|
|
|
else
|
|
|
qurey_test_sets = TestSet.find_by_sql("SELECT t.is_public,t.input, t.output,t.position FROM games g ,challenges c,test_sets t where
|
|
|
g.id=#{game.id} and c.id= g.challenge_id and t.challenge_id = c.id
|
|
|
")
|
|
|
end
|
|
|
|
|
|
|
|
|
# 能进入到此处,肯定是已经返回了结果,也就是说outputs中肯定有了数据
|
|
|
|
|
|
test_sets_count = qurey_test_sets.size
|
|
|
# had_test = Output.where(:game_id => game.id, :query_index => max_query_index)
|
|
|
# had_test_count = had_test.count
|
|
|
had_passed_testsests_error_count = max_query_index > 0 ? qurey_test_sets.select{|qurey_test_set| qurey_test_set.result == false}.count : 0
|
|
|
had_passed_testsests_error_count = test_sets_count - had_passed_testsests_error_count
|
|
|
|
|
|
had_test = Output.where(:game_id => game.id, :query_index => max_query_index)
|
|
|
error_position = had_test.blank? ? nil : had_test.select{|had_test| had_test.result == false}.last
|
|
|
|
|
|
Rails.logger.info("latest output id is #{error_position.id unless error_position.blank?}")
|
|
|
latest_output = error_position.try(:out_put).gsub(/\n/, '<br/>').gsub(/\t/, " \; \; \; \; \; \; \; \;") if error_position.try(:out_put).present?
|
|
|
Rails.logger.warn("last_output is #{latest_output}")
|
|
|
|
|
|
# power判断用户是否有权限查看隐藏测试集(TPM管理员;平台认证的老师;花费金币查看者)
|
|
|
power = (shixun_manager(shixun, current_user) || (current_user.is_certification_teacher)) ? 1 : (shixun.test_set_permission ? 0 : -1)
|
|
|
# 测试集统计及处理
|
|
|
unless qurey_test_sets.blank?
|
|
|
check_power = (power == 1 || game.test_sets_view)
|
|
|
test_sets = test_set_static_data(qurey_test_sets, check_power)
|
|
|
end
|
|
|
# 处理生成图片类型文件
|
|
|
picture = (game.picture_path.nil? ? 0 : game.id)
|
|
|
# 针对web类型的实训
|
|
|
web_route = game_challenge.try(:web_route)
|
|
|
mirror_name = shixun.mirror_name
|
|
|
|
|
|
# 轮询结束,更新评测耗时
|
|
|
e_record = EvaluateRecord.where(:game_id => game.id).first
|
|
|
if game_status == 0 || game_status == 2
|
|
|
if e_record
|
|
|
front_js = format("%.3f", (Time.now.to_f - e_record.try(:updated_at).to_f)).to_f
|
|
|
consume_time = format("%.3f", (Time.now - e_record.created_at)).to_f
|
|
|
e_record.update_attributes(:consume_time => consume_time, :front_js => front_js)
|
|
|
end
|
|
|
end
|
|
|
|
|
|
Rails.logger.warn("##game is is #{game.id}, record id is #{e_record.id}, time is**** #{Time.now.strftime("%Y-%m-%d %H:%M:%S.%L")}")
|
|
|
# 记录前端总耗时
|
|
|
record = e_record.try(:consume_time)
|
|
|
|
|
|
# 实训制作者当前拥有的金币
|
|
|
grade = User.where(:id => game.user_id).pluck(:grade).first
|
|
|
|
|
|
# 高性能取上一关、下一关
|
|
|
prev_game = Game.prev_identifier(shixun.id, game.myshixun_id, game_challenge.position)
|
|
|
next_game = Game.next_game(shixun.id, game.myshixun_id, game_challenge.position).try(:identifier)
|
|
|
|
|
|
output_hash = {:test_sets => test_sets, :had_test_count => test_sets_count, :test_sets_count => test_sets_count, :had_passed_testsests_error_count => had_passed_testsests_error_count}
|
|
|
|
|
|
return {:grade => grade, :gold => score, :experience => experience, :status => game_status, :had_done => had_done,
|
|
|
:position => game_challenge.position, :port => port, :power => power, :record => record,
|
|
|
:mirror_name => mirror_name, :picture => picture, :web_route => web_route, :latest_output => latest_output,
|
|
|
:star => game.star, :next_game => next_game, :prev_game => prev_game}.merge(output_hash)
|
|
|
end
|
|
|
|
|
|
# 记录实训花费的时间,前端是通过ent_time - open_time,所以最终只更新open_time即可
|
|
|
# 总花费的时间是花费的总时间加上开启的时间(open_time)
|
|
|
# 注意:endtime和当前时间没关系,非时间差值,必须和opentime结合使用才有意义
|
|
|
def cost_time params
|
|
|
game = Game.select([:id, :cost_time, :identifier]).find_by_identifier(params[:identifier])
|
|
|
cost_time = params[:time].to_i
|
|
|
game.update_attribute(:cost_time, cost_time)
|
|
|
return{ :game => game}
|
|
|
end
|
|
|
|
|
|
def sync_modify_time params
|
|
|
game = Game.select([:id, :modify_time, :challenge_id]).find_by_identifier(params[:identifier])
|
|
|
modify_time = Challenge.where(:id => game.challenge_id).pluck(:modify_time).first
|
|
|
game.update_column(:modify_time, modify_time)
|
|
|
return {:status => 1, :message => "success"}
|
|
|
end
|
|
|
|
|
|
# tpi弹框状态更新;true则不再显示;false每次刷新显示
|
|
|
def system_update params
|
|
|
myshixun = Myshixun.find(params[:myshixun_id])
|
|
|
myshixun.update_attribute(:system_tip, true)
|
|
|
return {:status => 1, :message => "success"}
|
|
|
end
|
|
|
|
|
|
# For admin
|
|
|
# 强制重置实训
|
|
|
def reset_my_game params, current_user
|
|
|
unless current_user.admin?
|
|
|
return {:status => 403, :message => "unauthorized"}
|
|
|
end
|
|
|
myshixun = Myshixun.find(Game.where(:identifier => params[:identifier]).pluck(:myshixun_id).first)
|
|
|
shixun = myshixun.shixun
|
|
|
ActiveRecord::Base.transaction do
|
|
|
begin
|
|
|
g = Gitlab.client
|
|
|
shixun_tomcat = Redmine::Configuration['shixun_tomcat']
|
|
|
challenges = shixun.challenges
|
|
|
# 删除选择题用户记录
|
|
|
unless challenges.blank?
|
|
|
challenges.each do |challenge|
|
|
|
if challenge.st != 0
|
|
|
challenge.challenge_chooses.each do |choose|
|
|
|
user_output = choose.choose_outputs
|
|
|
unless user_output.blank?
|
|
|
user_output.delete
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
#end
|
|
|
myshixun_job = Base64.urlsafe_encode64("myshixun_#{myshixun.id}")
|
|
|
StudentWork.where(:myshixun_id => myshixun.id).update_all(:myshixun_id => 0, :work_status => 0, :work_score => nil, :final_score => nil,
|
|
|
:cost_time => 0, :update_time => nil, :compelete_status => 0, :commit_time => nil)
|
|
|
# myshixun.destroy
|
|
|
myshixun.delete
|
|
|
# 主从复制出现脏读的情况
|
|
|
if myshixun.gpid
|
|
|
g.delete_project(myshixun.gpid)
|
|
|
end
|
|
|
rescue Exception => e
|
|
|
Rails.logger.error("myshixun reset failed #{e}")
|
|
|
raise ActiveRecord::Rollback
|
|
|
end
|
|
|
end
|
|
|
# 主从复制出现脏读的情况
|
|
|
sleep(1)
|
|
|
return {:status => 1, :message => "sucess", :url => "/shixuns/#{shixun.identifier}"}
|
|
|
end
|
|
|
|
|
|
def close_webssh params, current_user
|
|
|
myshixun_id = Game.where(:identifier => params[:identifier]).pluck(:myshixun_id).first
|
|
|
digest = params[:identifier]+Redmine::Configuration['bridge_secret_key']
|
|
|
digest_key = Digest::SHA1.hexdigest("#{digest}")
|
|
|
begin
|
|
|
shixun_tomcat = Redmine::Configuration['shixun_tomcat']
|
|
|
uri = "#{shixun_tomcat}/bridge/webssh/delete"
|
|
|
Rails.logger.info("#{current_user} => cloese_webssh digest is #{digest}")
|
|
|
params = {:tpiID => myshixun_id, :digestKey => digest_key, :identifier => params[:identifier]}
|
|
|
res = uri_exec uri, params
|
|
|
if res && res['code'].to_i != 0
|
|
|
raise("实训云平台繁忙(繁忙等级:110)")
|
|
|
end
|
|
|
return {:status => 1, :message => "webssh closed"}
|
|
|
rescue Exception => e
|
|
|
Rails.logger.error(e)
|
|
|
return {:status => 0, :message => "authentication failed"}
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# return: 头像,姓名,用户主页链接,学校, 点赞数, 等级(通关数)
|
|
|
def get_passed_user params
|
|
|
page = params[:page].to_i
|
|
|
offset = page * 15
|
|
|
file_base params[:identifier]
|
|
|
games = @game.challenge.games.where(:status => 2).includes(user: [:myshixuns, user_extensions: :school]).order("cost_time desc").offset(offset).limit(15)
|
|
|
users = format_answer_list games
|
|
|
{user_answer_list: users}
|
|
|
end
|
|
|
|
|
|
def get_passed_code params
|
|
|
file_base params[:identifier]
|
|
|
codes = @game.game_codes.select([:new_code, :path]).where("new_code is not null")
|
|
|
code_list = []
|
|
|
codes.each do |code|
|
|
|
code_list << {path: code.path, code: code.new_code}
|
|
|
end
|
|
|
{code_list: code_list}
|
|
|
end
|
|
|
|
|
|
|
|
|
def shixun_manager shixun, user_current
|
|
|
member = shixun.shixun_members.where(:role => [1,2], :user_id => user_current.id)
|
|
|
(!member.blank? || user_current.admin?) ? true : false
|
|
|
end
|
|
|
|
|
|
# 实训一些基础查询
|
|
|
def game_base identifier
|
|
|
@game = Game.select([:myshixun_id, :status, :final_score, :answer_open, :challenge_id, :id]).find_by_identifier(identifier)
|
|
|
@myshixun = Myshixun.select([:id, :shixun_id, :gpid, :user_id]).find(@game.myshixun_id)
|
|
|
@shixun = Shixun.select([:status, :id]).find(@myshixun.shixun_id)
|
|
|
end
|
|
|
|
|
|
# 与文件代码相关的一些通用查询
|
|
|
def file_base identifier
|
|
|
@game = Game.select([:myshixun_id, :challenge_id, :id]).find_by_identifier(identifier)
|
|
|
@myshixun = Myshixun.select([:shixun_id, :gpid, :id, :user_id]).find(@game.myshixun_id)
|
|
|
end
|
|
|
|
|
|
private
|
|
|
|
|
|
def format_answer_list games
|
|
|
user_info = []
|
|
|
games.each do |game|
|
|
|
user = game.user
|
|
|
votes = game.game_codes.first.votes
|
|
|
passed_count = user.myshixuns.where(:status => 1).count
|
|
|
user_info << {
|
|
|
user_id: user.id,
|
|
|
name: user.show_name,
|
|
|
game_id: game.id,
|
|
|
image_url: url_to_avatar(user),
|
|
|
school_name: user.school_name,
|
|
|
votes: votes,
|
|
|
passed_count: passed_count
|
|
|
}
|
|
|
end
|
|
|
user_info
|
|
|
end
|
|
|
end
|