forked from jasder/forgeplus
283 lines
9.2 KiB
Ruby
283 lines
9.2 KiB
Ruby
|
class HacksController < ApplicationController
|
|||
|
before_action :require_login, except: [:index]
|
|||
|
before_action :find_hack, only: [:edit, :update, :publish, :start, :update_set, :delete_set, :destroy, :cancel_publish]
|
|||
|
before_action :require_teacher_identity, only: [:create]
|
|||
|
before_action :require_auth_identity, only: [:update, :edit, :publish, :update_set, :delete_set, :destroy, :cancel_publish]
|
|||
|
|
|||
|
|
|||
|
# 开启编程,如果第一次开启,创建一条记录,如果已经开启过的话,直接返回标识即可
|
|||
|
def start
|
|||
|
# 未发布的编程题,只能作者、或管理员访问
|
|||
|
start_hack_auth
|
|||
|
user_hack = @hack.hack_user_lastest_codes.where(user_id: current_user.id).first
|
|||
|
logger.info("#user_hack: #{user_hack}")
|
|||
|
identifier =
|
|||
|
if user_hack.present?
|
|||
|
logger.info("#####user_hack_id:#{user_hack.id}")
|
|||
|
user_hack.identifier
|
|||
|
else
|
|||
|
user_identifier = generate_identifier HackUserLastestCode, 12
|
|||
|
user_code = {user_id: current_user.id, code: @hack.code, modify_time: Time.now,
|
|||
|
identifier: user_identifier, language: @hack.language}
|
|||
|
@hack.hack_user_lastest_codes.create!(user_code)
|
|||
|
user_identifier
|
|||
|
end
|
|||
|
render_ok(identifier: identifier)
|
|||
|
end
|
|||
|
|
|||
|
# 首页
|
|||
|
def index
|
|||
|
# 筛选过滤与排序
|
|||
|
params_filter_or_order
|
|||
|
# 我解决的编程题数
|
|||
|
user_codes = HackUserLastestCode.joins(:hack).mine_hack(current_user).passed
|
|||
|
@simple_count = user_codes.where(hacks: {difficult: 1}).count
|
|||
|
@medium_count = user_codes.where(hacks: {difficult: 2}).count
|
|||
|
@diff_count = user_codes.where(hacks: {difficult: 3}).count
|
|||
|
@pass_count = @simple_count + @medium_count + @diff_count
|
|||
|
|
|||
|
@hacks_count = @hacks.count("hacks.id")
|
|||
|
@hacks = paginate @hacks
|
|||
|
end
|
|||
|
|
|||
|
def create
|
|||
|
begin
|
|||
|
logger.info("##########{hack_params}")
|
|||
|
tip_exception("一次只能增加50个测试集") if hack_sets_params.size > 50
|
|||
|
tip_exception("一次只能增加50个知识点") if params[:tags].size > 50
|
|||
|
hack = Hack.new(hack_params)
|
|||
|
hack.user_id = current_user.id
|
|||
|
hack.identifier = generate_identifier Hack, 8
|
|||
|
tag_params = params[:tags].map{|tag| {tag_discipline_id: tag}}
|
|||
|
ActiveRecord::Base.transaction do
|
|||
|
hack.save!
|
|||
|
# 创建测试集与代码
|
|||
|
hack.hack_sets.create!(hack_sets_params)
|
|||
|
# 新建知识点
|
|||
|
hack.tag_discipline_containers.create!(tag_params) if tag_params.present?
|
|||
|
hack_codes = hack.hack_codes.new(hack_code_params)
|
|||
|
hack_codes.modify_time = Time.now
|
|||
|
hack_codes.save!
|
|||
|
new_item_params = item_params.merge(container: hack, item_type: 'PROGRAM', difficulty: params[:hack][:difficult], user_id: current_user.id)
|
|||
|
ItemBank.create!(new_item_params)
|
|||
|
end
|
|||
|
render_ok({identifier: hack.identifier})
|
|||
|
rescue => e
|
|||
|
logger.error("########create_hack_error: #{e.message}")
|
|||
|
render_error("创建失败: #{e.message}")
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def update
|
|||
|
begin
|
|||
|
# 知识点
|
|||
|
tag_discipline_ids = @hack.tag_discipline_containers.pluck(:tag_discipline_id)
|
|||
|
new_tag_ids = params[:tags].to_a - tag_discipline_ids
|
|||
|
tag_params = new_tag_ids.map{|tag| {tag_discipline_id: tag}}
|
|||
|
ActiveRecord::Base.transaction do
|
|||
|
@hack.update_attributes!(hack_params)
|
|||
|
set_ids = @hack.hack_sets.pluck(:id)
|
|||
|
# 更新
|
|||
|
param_update_sets params[:update_hack_sets], set_ids
|
|||
|
# 新建
|
|||
|
@hack.hack_sets.create!(hack_sets_params)
|
|||
|
# 更新代码
|
|||
|
code_params = params[:hack_codes][:code] != @hack.code ? hack_code_params.merge(modify_time: Time.now) : hack_code_params
|
|||
|
@hack.hack_codes.first.update_attributes!(code_params)
|
|||
|
@hack.tag_discipline_containers.create!(tag_params) if tag_params
|
|||
|
@hack.tag_discipline_containers.where.not(tag_discipline_id: params[:tags]).destroy_all
|
|||
|
|
|||
|
# 更新题库相关记录
|
|||
|
if @hack.item_bank.present?
|
|||
|
update_item_params = item_params.merge({difficulty: params[:hack][:difficult]})
|
|||
|
@hack.item_bank.update!(update_item_params)
|
|||
|
end
|
|||
|
end
|
|||
|
render_ok
|
|||
|
rescue Exception => e
|
|||
|
logger.error("####update_hack_error: #{e.message}")
|
|||
|
render_error("更新失败: #{e.message}")
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
# 更新测试集接口
|
|||
|
def update_set
|
|||
|
set = @hack.hack_sets.find_by(id: params[:id])
|
|||
|
set.update_attributes!(hack_set_params)
|
|||
|
render_ok
|
|||
|
end
|
|||
|
|
|||
|
# 单独删除测试集
|
|||
|
def delete_set
|
|||
|
set = @hack.hack_sets.find_by(id: params[:id])
|
|||
|
set.destroy!
|
|||
|
render_ok
|
|||
|
end
|
|||
|
|
|||
|
# 发布功能
|
|||
|
def publish
|
|||
|
@hack.update_attribute(:status, 1)
|
|||
|
base_attrs = {
|
|||
|
trigger_user_id: current_user.id, viewed: 0, tiding_type: 'System', user_id: @hack.user_id,
|
|||
|
parent_container_type: "HackPublish", extra: @hack.identifier
|
|||
|
}
|
|||
|
@hack.tidings.create!(base_attrs)
|
|||
|
render_ok
|
|||
|
end
|
|||
|
|
|||
|
# 取消发布
|
|||
|
def cancel_publish
|
|||
|
@hack.update_attribute(:status, 0)
|
|||
|
base_attrs = {
|
|||
|
trigger_user_id: current_user.id, viewed: 0, tiding_type: 'System', user_id: @hack.user_id,
|
|||
|
parent_container_type: "HackUnPublish", extra: @hack.identifier
|
|||
|
}
|
|||
|
@hack.tidings.create!(base_attrs)
|
|||
|
render_ok
|
|||
|
end
|
|||
|
|
|||
|
# 发布列表
|
|||
|
def unpulished_list
|
|||
|
limit = params[:limit] || 16
|
|||
|
page = params[:page] || 1
|
|||
|
hacks = Hack.where(user_id: current_user.id, status: 0)
|
|||
|
@hacks_count = hacks.count
|
|||
|
@hacks = hacks.includes(:hack_sets).page(page).per(limit)
|
|||
|
end
|
|||
|
|
|||
|
def edit;
|
|||
|
end
|
|||
|
|
|||
|
def new;
|
|||
|
end
|
|||
|
|
|||
|
def destroy
|
|||
|
begin
|
|||
|
base_attrs = {
|
|||
|
user_id: @hack.user_id, viewed: 0, tiding_type: 'System', trigger_user_id: current_user.id,
|
|||
|
parent_container_type: "HackDelete", extra: "#{@hack.name}"
|
|||
|
}
|
|||
|
@hack.tidings.create!(base_attrs)
|
|||
|
@hack.destroy
|
|||
|
render_ok
|
|||
|
rescue => e
|
|||
|
logger.error("####hack_delete_error: #{e.message}")
|
|||
|
render_error("删除失败")
|
|||
|
end
|
|||
|
|
|||
|
end
|
|||
|
|
|||
|
private
|
|||
|
|
|||
|
# 实名认证老师,管理员与运营人员权限
|
|||
|
def require_teacher_identity
|
|||
|
unless current_user.certification_teacher? || admin_or_business?
|
|||
|
tip_exception(403, "..")
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
# 只有自己,或者管理员才能更新
|
|||
|
def require_auth_identity
|
|||
|
unless @hack.user_id == current_user.id || admin_or_business?
|
|||
|
tip_exception(403, "..")
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def find_hack
|
|||
|
@hack = Hack.find_by_identifier(params[:identifier])
|
|||
|
end
|
|||
|
|
|||
|
def hack_params
|
|||
|
params.require(:hack).permit(:name, :description, :difficult, :open_or_not, :time_limit, :score, :sub_discipline_id)
|
|||
|
end
|
|||
|
|
|||
|
def item_params
|
|||
|
params.require(:hack).permit(:name, :sub_discipline_id)
|
|||
|
end
|
|||
|
|
|||
|
def hack_sets_params
|
|||
|
params.permit(hack_sets: [:input, :output, :position])[:hack_sets]
|
|||
|
end
|
|||
|
|
|||
|
def hack_set_params
|
|||
|
params.require(:hack_set).permit(:id, :input, :output, :position)
|
|||
|
end
|
|||
|
|
|||
|
def hack_code_params
|
|||
|
params.require(:hack_codes).permit(:code, :language)
|
|||
|
end
|
|||
|
|
|||
|
def publish_params
|
|||
|
params.require(:hack).permit(:difficult, :category, :open_or_not, :time_limit, :score)
|
|||
|
end
|
|||
|
|
|||
|
def param_update_sets sets, all_sets_id
|
|||
|
delete_set_ids = all_sets_id - sets.map {|set| set[:id]}
|
|||
|
@hack.hack_sets.where(id: delete_set_ids).destroy_all
|
|||
|
logger.info("#######sets:#{sets}")
|
|||
|
sets.each do |set|
|
|||
|
logger.info("###set[:id]: #{set[:id]}")
|
|||
|
logger.info("###all_sets: #{all_sets_id.include?(set[:id])}")
|
|||
|
if all_sets_id.include?(set[:id])
|
|||
|
update_attrs = {input: set[:input], output: set[:output], position: set[:position]}
|
|||
|
@hack.hack_sets.find_by!(id: set[:id]).update_attributes!(update_attrs)
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def params_filter_or_order
|
|||
|
# 如果有来源,就不管发布公开私有
|
|||
|
select_sql = "hacks.*, if(hacks.hack_user_lastest_codes_count=0, 0, hacks.pass_num/hacks.hack_user_lastest_codes_count) passed_rate"
|
|||
|
if params[:come_from]
|
|||
|
hacks = Hack.select(select_sql).mine(current_user.id)
|
|||
|
else
|
|||
|
# 全部包括已经发布的,和我的未发布的
|
|||
|
if current_user.admin_or_business?
|
|||
|
hacks = Hack.select(select_sql)
|
|||
|
else
|
|||
|
hacks = Hack.select(select_sql).published.opening.or(Hack.select(select_sql).unpublish.mine(current_user.id))
|
|||
|
end
|
|||
|
end
|
|||
|
# 搜索
|
|||
|
if params[:search]
|
|||
|
hacks = hacks.where("name like ?", "%#{params[:search]}%")
|
|||
|
end
|
|||
|
# 难度
|
|||
|
if params[:difficult]
|
|||
|
hacks = hacks.where(difficult: params[:difficult])
|
|||
|
end
|
|||
|
# 状态
|
|||
|
if params[:status]
|
|||
|
user_hacks = HackUserLastestCode.where(user_id: current_user.id)
|
|||
|
if params[:status].to_i == -1
|
|||
|
if user_hacks.present?
|
|||
|
hacks = hacks.where.not(id: user_hacks.pluck(:hack_id))
|
|||
|
end
|
|||
|
else
|
|||
|
hacks = hacks.joins(:hack_user_lastest_codes).where(hack_user_lastest_codes: {status: params[:status]})
|
|||
|
end
|
|||
|
end
|
|||
|
# 分类
|
|||
|
if params[:category]
|
|||
|
hacks = hacks.where(category: params[:category])
|
|||
|
end
|
|||
|
|
|||
|
# 语言
|
|||
|
if params[:language]
|
|||
|
hacks = hacks.joins(:hack_codes).where(hack_codes: {language: params[:language]})
|
|||
|
end
|
|||
|
|
|||
|
# 排序
|
|||
|
sort_by = params[:sort_by] || "hack_user_lastest_codes_count"
|
|||
|
sort_direction = params[:sort_direction] || "desc"
|
|||
|
@hacks = hacks.order("#{sort_by} #{sort_direction}")
|
|||
|
end
|
|||
|
|
|||
|
def start_hack_auth
|
|||
|
return true if @hack.status == 1
|
|||
|
require_auth_identity
|
|||
|
end
|
|||
|
|
|||
|
|
|||
|
end
|