|
|
class ChallengesController < ApplicationController
|
|
|
before_action :require_login, :check_auth, except: [:index]
|
|
|
before_action :find_shixun, only: [:new, :create, :index]
|
|
|
skip_before_action :verify_authenticity_token, only: [:create, :update, :create_choose_question, :crud_answer]
|
|
|
before_action :find_challenge, only: [:edit, :show, :update, :create_choose_question, :index_down, :index_up,
|
|
|
:edit_choose_question, :show_choose_question, :destroy_challenge_choose,
|
|
|
:update_choose_question, :destroy, :crud_answer, :answer]
|
|
|
# 关卡更新和操作的权限控制
|
|
|
before_action :update_allowed, except: [:index]
|
|
|
# 关卡访问的权限控制
|
|
|
before_action :shixun_access_allowed, only: [:index]
|
|
|
|
|
|
include ShixunsHelper
|
|
|
include ChallengesHelper
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# 新建实践题
|
|
|
def new
|
|
|
@position = @shixun.challenges.count + 1
|
|
|
@st = params[:st].to_i
|
|
|
@task_pass_default = PlatformSample.find_by(samples_type: "taskPass").try(:contents)
|
|
|
end
|
|
|
|
|
|
# params
|
|
|
# challenge:{"subject": "标题", "task_pass": "过关任务",
|
|
|
# "diffculty": "关卡难度", "score": "关卡分数", "st": "关卡类型"} 关卡相关信息
|
|
|
# challenge_tag: 关卡标签
|
|
|
#
|
|
|
def create
|
|
|
ActiveRecord::Base.transaction do
|
|
|
begin
|
|
|
@challenge = Challenge.new(challenge_params)
|
|
|
@challenge.position = @shixun.challenges.count + 1
|
|
|
@challenge.shixun_id = @shixun.id
|
|
|
@challenge.user_id = current_user.id
|
|
|
@challenge.modify_time = Time.now
|
|
|
@challenge.save!
|
|
|
|
|
|
# 实训是否需要重置, 非实践任务创建第一个阶段调用, 避免不包含实践任务的实训进行模拟实战报错 todo: 新建实训需要重置TPI
|
|
|
# shixun_modify_status_without_publish(@shixun, 1) if @challenge.position != 1
|
|
|
|
|
|
tags = params[:challenge_tag]
|
|
|
if tags.present?
|
|
|
tags.each do |tag|
|
|
|
# TODO 创建tag的时候为什么一直报challenge choose必须存在??
|
|
|
ChallengeTag.create!(name: tag, challenge_id: @challenge.id)
|
|
|
end
|
|
|
end
|
|
|
rescue Exception => e
|
|
|
uid_logger_error("create challenge failed #{e}")
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 创建选择题
|
|
|
def create_choose_question
|
|
|
ActiveRecord::Base.transaction do
|
|
|
begin
|
|
|
@challenge_choose = ChallengeChoose.new(chooce_params)
|
|
|
@challenge_choose.position = @challenge.challenge_chooses.count + 1
|
|
|
@challenge_choose.category = @challenge_choose.standard_answer.length == 1 ? 1 : 2
|
|
|
@challenge_choose.challenge_id = @challenge.id
|
|
|
|
|
|
if @challenge_choose.save!
|
|
|
# 创建选项
|
|
|
params[:question][:cnt].each_with_index do |test, index|
|
|
|
answer = params[:choice][:answer][index]
|
|
|
ChallengeQuestion.create(:option_name => test,
|
|
|
:challenge_choose_id => @challenge_choose.id,
|
|
|
:position => index, :right_key => answer)
|
|
|
end
|
|
|
# 创建单选多选的技能标签
|
|
|
if params[:challenge_tag].present?
|
|
|
params[:challenge_tag].each do |tag|
|
|
|
ChallengeTag.create(:name => tag, :challenge_choose_id => @challenge_choose.id, :challenge_id => @challenge.id)
|
|
|
end
|
|
|
end
|
|
|
@challenge.update_column(:score, @challenge.challenge_chooses.sum(:score))
|
|
|
end
|
|
|
rescue Exception => e
|
|
|
raise ActiveRecord::Rollback
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 选择题详情页面
|
|
|
def show_choose_question
|
|
|
@challenge_choose = ChallengeChoose.find params[:choose_id]
|
|
|
|
|
|
end
|
|
|
|
|
|
# 选择题更新页面
|
|
|
def update_choose_question
|
|
|
@challenge_choose = ChallengeChoose.find(params[:choose_id])
|
|
|
ActiveRecord::Base.transaction do
|
|
|
if params[:standard_answer] != @challenge_choose.standard_answer
|
|
|
@challenge.update_column(:modify_time, Time.now)
|
|
|
end
|
|
|
@challenge_choose.update_attributes(chooce_params)
|
|
|
@challenge.update_column(:score, @challenge.challenge_chooses.sum(:score))
|
|
|
# 单选多选题的更新
|
|
|
category = @challenge_choose.standard_answer.length > 1 ? 2 : 1
|
|
|
@challenge_choose.update_column(:category, category)
|
|
|
begin
|
|
|
@challenge_choose.challenge_questions.delete_all
|
|
|
params[:question][:cnt].each_with_index do |test, index|
|
|
|
answer = params[:choice][:answer][index]
|
|
|
ChallengeQuestion.create(:option_name => test, :challenge_choose_id => @challenge_choose.id, :position => index, :right_key => answer)
|
|
|
end
|
|
|
@challenge_choose.challenge_tags.delete_all unless @challenge_choose.challenge_tags.blank?
|
|
|
if params[:challenge_tag].present?
|
|
|
params[:challenge_tag].each do |tag|
|
|
|
ChallengeTag.create(:name => tag, :challenge_choose_id => @challenge_choose.id, :challenge_id => @challenge.id)
|
|
|
end
|
|
|
end
|
|
|
@challenge_choose = ChallengeChoose.find params[:choose_id]
|
|
|
rescue Exception => e
|
|
|
raise ActiveRecord::Rollback
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 选择题的编辑
|
|
|
def edit_choose_question
|
|
|
@challenge_choose = ChallengeChoose.find params[:choose_id]
|
|
|
end
|
|
|
|
|
|
def destroy_challenge_choose
|
|
|
ActiveRecord::Base.transaction do
|
|
|
@challenge_choose = ChallengeChoose.where(:id => params[:choose_id]).first
|
|
|
pos = @challenge_choose.position
|
|
|
@challenge.challenge_chooses.where("position > ?", pos).update_all("position = position - 1")
|
|
|
@challenge_choose.destroy
|
|
|
@status = 1
|
|
|
# 发起重置请求 TODO: 重置实训需要后续做
|
|
|
# shixun_modify_status_without_publish(@shixun, 1)
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 编辑模式
|
|
|
# tab 0,nil 过关任务, 1 评测设置, 2 参考答案
|
|
|
def edit
|
|
|
@tab = params[:tab].to_i
|
|
|
@power = @shixun.status == 0
|
|
|
challenge_num = Challenge.where(:shixun_id => @shixun).count
|
|
|
@position = @challenge.position
|
|
|
@chooses = @challenge.challenge_chooses
|
|
|
if @position < challenge_num
|
|
|
@next_challenge = Challenge.where(:shixun_id => @shixun, :position => @position + 1).first
|
|
|
end
|
|
|
@prev_challenge = Challenge.where(:shixun_id => @shixun, :position => @position - 1).first if @position - 1 > 0
|
|
|
end
|
|
|
|
|
|
def index
|
|
|
uid_logger("identifier: #{params}")
|
|
|
base_columns = "challenges.id, challenges.subject, challenges.st, challenges.score, challenges.position,
|
|
|
challenges.shixun_id, games.identifier, games.status"
|
|
|
join_sql = "LEFT JOIN games ON games.challenge_id = challenges.id AND games.user_id = #{current_user.id}"
|
|
|
# 下面2个参数是为了解决列表获取通关人数与正在游玩人数的问题
|
|
|
#@pass_games_map = @shixun.challenges.joins(:games).where(games: {status:2}).group(:challenge_id).reorder(nil).count
|
|
|
#@play_games_map = @shixun.challenges.joins(:games).where(games: {status:[0,1]}).group(:challenge_id).reorder(nil).count
|
|
|
|
|
|
@challenges = @shixun.challenges.joins(join_sql).select(base_columns).uniq
|
|
|
|
|
|
#@challenges = @shixun.challenges.fields_for_list
|
|
|
@editable = @shixun.status == 0 # before_action:有判断权限,如果没发布,则肯定是管理人员
|
|
|
@user = current_user
|
|
|
@shixun.increment!(:visits)
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
def show
|
|
|
@tab = params[:tab].nil? ? 1 : params[:tab].to_i
|
|
|
challenge_num = @shixun.challenges_count
|
|
|
@power = @shixun.status == 0 # 之前验证走过了是不是管理员,因此这里只用判断是否发布
|
|
|
@position = @challenge.position
|
|
|
if @position < challenge_num
|
|
|
@next_challenge = Challenge.where(:shixun_id => @shixun, :position => @position + 1).first
|
|
|
end
|
|
|
@prev_challenge = Challenge.where(:shixun_id => @shixun, :position => @position - 1).first if @position - 1 > 0
|
|
|
end
|
|
|
|
|
|
# tab 0:过关任务的更新; 1:评测设置的更新; 2:表示参考答案的更新;
|
|
|
def update
|
|
|
begin
|
|
|
ActiveRecord::Base.transaction do
|
|
|
tab = params[:tab].to_i
|
|
|
@challenge.update_attributes!(challenge_params)
|
|
|
if tab == 0 && @challenge.st == 0
|
|
|
@challenge.challenge_tags.delete_all
|
|
|
if params[:challenge_tag].present?
|
|
|
params[:challenge_tag].each do |input|
|
|
|
ChallengeTag.create!(:name => input, :challenge_id => @challenge.id)
|
|
|
end
|
|
|
end
|
|
|
elsif tab == 1
|
|
|
path = @challenge.path
|
|
|
exec_path = @challenge.exec_path
|
|
|
test_set = @challenge.test_sets
|
|
|
sets_output = test_set.map(&:output)
|
|
|
sets_input = test_set.map(&:input)
|
|
|
sets_open = test_set.map(&:is_public)
|
|
|
set_score = test_set.map(&:score)
|
|
|
set_match_rule = test_set.map(&:match_rule)
|
|
|
params_hidden = params[:test_set].map{|set| set[:hidden].to_i == 0}
|
|
|
params_output = params[:test_set].map{|set| set[:output] }
|
|
|
params_input = params[:test_set].map{|set| set[:input] }
|
|
|
params_score = params[:test_set].map{|set| set[:score]}
|
|
|
params_test_set = params[:test_set].map{|set| set[:match_rule]}
|
|
|
# 测试集变化则需要更新(输入、 输出、 是否隐藏)
|
|
|
if sets_output != params_output || sets_open != params_hidden || sets_input != params_input ||
|
|
|
set_score != params_score || params_test_set != set_match_rule
|
|
|
test_set.delete_all unless test_set.blank?
|
|
|
params[:test_set].each_with_index do |set, index|
|
|
|
# last: 末尾匹配, full: 全完匹配
|
|
|
logger.info("set: #{set}; match_rule : #{set[:match_rule]}")
|
|
|
match_rule = set[:match_rule] == 'last' ? 'last' : 'full'
|
|
|
TestSet.create!(:challenge_id => @challenge.id,
|
|
|
:input => "#{set[:input]}",
|
|
|
:output => "#{set[:output]}",
|
|
|
:is_public => params_hidden[index],
|
|
|
:score => set[:score],
|
|
|
:match_rule => "#{match_rule}",
|
|
|
:position => (index + 1))
|
|
|
end
|
|
|
@challenge.update_column(:modify_time, Time.now)
|
|
|
# 测试集的
|
|
|
@shixun.myshixuns.update_all(:system_tip => 0)
|
|
|
end
|
|
|
if params[:challenge][:show_type].to_i == -1
|
|
|
@challenge.update_attributes(picture_path: nil, web_route: nil, expect_picture_path: nil, original_picture_path: nil)
|
|
|
end
|
|
|
# 关卡评测执行文件如果被修改,需要修改脚本内容
|
|
|
logger.info("############shixun_publiced:#{@shixun.public == 0}")
|
|
|
if @shixun.public == 0
|
|
|
script = modify_shixun_script @shixun, @shixun.evaluate_script
|
|
|
@shixun.shixun_info.update_column(:evaluate_script, script) if script.present?
|
|
|
end
|
|
|
# TODO:
|
|
|
# if path != params[:challenge][:path]
|
|
|
# shixun_modify_status_without_publish(@shixun, 1)
|
|
|
# end
|
|
|
#Attachment.attach_files(@challenge, params[:attachments])
|
|
|
end
|
|
|
|
|
|
end
|
|
|
rescue Exception => e
|
|
|
logger.error("##update_challenges: ##{e.message}")
|
|
|
tip_exception("#{e.message}")
|
|
|
end
|
|
|
|
|
|
end
|
|
|
|
|
|
# 参考答案的'增,删,改'
|
|
|
# POST: /shixuns/:id/challenges/:id/crud_answer
|
|
|
# {'challenge_answer': [
|
|
|
# {'name': 'name', contents: 'contents', score: 10},
|
|
|
# {...},
|
|
|
# {...}, ...]
|
|
|
#}
|
|
|
def crud_answer
|
|
|
if @challenge.challenge_answers && params[:challenge_answer].blank?
|
|
|
@challenge.challenge_answers.destroy_all
|
|
|
else
|
|
|
raise '参考答案不能为空' if params[:challenge_answer].empty?
|
|
|
raise '占比之和必须为100%' if params[:challenge_answer].map{|a| a[:score]}.sum != 100
|
|
|
ActiveRecord::Base.transaction do
|
|
|
@challenge.challenge_answers.destroy_all if @challenge.challenge_answers
|
|
|
params[:challenge_answer].each_with_index do |answer, index|
|
|
|
# 内容为空不保存
|
|
|
next if answer[:contents].blank?
|
|
|
ChallengeAnswer.create!(name: answer[:name], contents: answer[:contents],
|
|
|
level: index+1, score: answer[:score], challenge_id: @challenge.id)
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
end
|
|
|
|
|
|
# 查看参考答案接口
|
|
|
def answer
|
|
|
@answers = @challenge.challenge_answers
|
|
|
end
|
|
|
|
|
|
def index_down
|
|
|
next_challenge = @challenge.next_challenge
|
|
|
position = @challenge.position
|
|
|
begin
|
|
|
ActiveRecord::Base.transaction do
|
|
|
@challenge.update_attributes!(position: (position + 1))
|
|
|
next_challenge.update_attributes!(position: next_challenge.position - 1)
|
|
|
# 关卡位置被修改,需要修改脚本
|
|
|
script = modify_shixun_script @shixun, @shixun.evaluate_script
|
|
|
@shixun.shixun_info.update_attributes!(evaluate_script: script)
|
|
|
end
|
|
|
rescue => e
|
|
|
tip_exception("下移失败: #{e.message}")
|
|
|
end
|
|
|
end
|
|
|
|
|
|
def index_up
|
|
|
position = @challenge.position
|
|
|
last_challenge = @challenge.last_challenge
|
|
|
begin
|
|
|
ActiveRecord::Base.transaction do
|
|
|
@challenge.update_attribute(:position, (position - 1))
|
|
|
last_challenge.update_attribute(:position, last_challenge.position + 1)
|
|
|
# 关卡位置被修改,需要修改脚本
|
|
|
script = modify_shixun_script @shixun, @shixun.evaluate_script
|
|
|
@shixun.shixun_info.update_column(:evaluate_script, script)
|
|
|
end
|
|
|
rescue => e
|
|
|
tip_exception("上移失败: #{e.message}")
|
|
|
end
|
|
|
|
|
|
end
|
|
|
|
|
|
def destroy
|
|
|
next_challenges = @shixun.challenges.where("position > #{@challenge.position}")
|
|
|
next_challenges.update_all("position = position - 1")
|
|
|
# Todo: 实训修改后,关卡需要重置
|
|
|
# shixun_modify_status_without_publish(@shixun, 1)
|
|
|
@challenge.destroy
|
|
|
# 关卡位置被删除,需要修改脚本
|
|
|
script = modify_shixun_script @shixun, @shixun.evaluate_script
|
|
|
@shixun.shixun_info.update_column(:evaluate_script, script)
|
|
|
end
|
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
def find_shixun
|
|
|
@shixun = Shixun.find_by_identifier(params[:shixun_identifier])
|
|
|
end
|
|
|
|
|
|
# 通用接口
|
|
|
def find_challenge
|
|
|
@challenge = Challenge.find params[:id]
|
|
|
@shixun = Shixun.find_by!(identifier: params[:shixun_identifier])
|
|
|
end
|
|
|
|
|
|
def challenge_params
|
|
|
tip_exception("评测时间不能超过300秒") if params[:challenge][:exec_time].to_i > 300
|
|
|
params.require(:challenge).permit(:subject, :task_pass, :difficulty, :score, :st, :modify_time, :test_set_average,
|
|
|
:path, :exec_path, :show_type, :original_picture_path, :test_set_score,
|
|
|
:expect_picture_path, :picture_path, :web_route, :answer, :exec_time)
|
|
|
end
|
|
|
|
|
|
def chooce_params
|
|
|
params.require(:challenge_choose).permit(:subject, :answer,
|
|
|
:standard_answer, :score, :difficult)
|
|
|
end
|
|
|
|
|
|
def update_allowed
|
|
|
unless current_user.manager_of_shixun?(@shixun)
|
|
|
raise Educoder::TipException.new(403, "..")
|
|
|
end
|
|
|
end
|
|
|
|
|
|
end
|