# encoding: utf-8
# Redmine - project management software
# Copyright (C) 2006-2013 Jean-Philippe Lang
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
# Time 2015-01-28 16:34:21
# Author lizanle
# Description 封装代码,简化代码,格式化代码,
class ProjectsController < ApplicationController
layout 'base_projects'
before_filter :authorize1 , :only = > [ :show ]
# menu_item :overview, :only => :show
# menu_item :roadmap, :only => :roadmap
# menu_item :settings, :only => :settings
# menu_item :homework, :only => [:homework, :new_homework]
# menu_item :feedback, :only => :feedback
# menu_item :share, :only => :share
skip_before_filter :verify_authenticity_token , :only = > [ :training_task_status ]
skip_before_filter :check_if_login_required , :only = > [ :training_task_status ]
before_filter :check_authentication
before_filter :find_project , :except = > [ :index , :search , :list , :new , :create , :copy , :statistics , :new_join , :course , :enterprise_course , :course_enterprise ,
:view_homework_attaches , :join_project , :project_home , :training_execute , :training_task_status ]
before_filter :authorize , :only = > [ :show , :settings , :edit , :sort_project_members , :update , :modules , :close , :reopen , :view_homework_attaches , :course ]
before_filter :authorize_global , :only = > [ :new , :create , :view_homework_attaches ]
before_filter :require_admin , :only = > [ :copy , :unarchive , :calendar ]
before_filter :require_admin_or_manager , :only = > [ :destroy ]
before_filter :file
# 除非项目内人员,不可查看成员, TODO: 完了写报表里去
# before_filter :memberAccess, only: :member
# accept_rss_auth :index
accept_api_auth :index , :show , :create , :update , :destroy
after_filter :only = > [ :create , :edit , :update , :archive , :unarchive , :destroy ] do | controller |
if controller . request . post?
controller . send :expire_action , :controller = > 'welcome' , :action = > 'robots'
end
end
helper :bids
include BidsHelper
helper :contests
include ContestsHelper
helper :sort
include SortHelper
helper :custom_fields
include CustomFieldsHelper
helper :issues
helper :queries
include QueriesHelper
helper :repositories
include RepositoriesHelper
include ProjectsHelper
helper :members
helper :activities
helper :documents
helper :watchers
# helper :watcherlist
helper :words
helper :project_score
helper :user_score
include UsersHelper
### added by william
include ActsAsTaggableOn :: TagsHelper
include ApplicationHelper
DCODES = %W( 2 3 4 5 6 7 8 9 a b c f e f g h i j k l m n o p q r s t u v w x y z )
# 仅仅为了转换Gitlab地址
def project_home
rep = params [ :rep ]
login = params [ :username ]
begin
user = User . find_by_login ( login )
project = Project . find_by_sql ( " SELECT projects.* FROM `repositories`,`projects` where repositories.project_id = projects.id and projects.user_id = #{ user . try ( :id ) } and repositories.identifier=' #{ rep } ' " ) . first
respond_to do | format |
format . html { redirect_to ( :controller = > 'repositories' , :action = > 'show' , :id = > project . id , :repository_id = > rep ) }
end
rescue
render_404
return
end
end
#查找组织
def search_public_orgs_not_in_project
condition = '%%'
if ! params [ :name ] . nil?
condition = " % #{ params [ :name ] . strip } % " . gsub ( " " , " " )
end
limit = 15
project_org_ids = OrgProject . find_by_sql ( " select distinct organization_id from org_projects where project_id = #{ params [ :id ] } " ) . map ( & :organization_id ) << 0
@orgs_not_in_project = User . current . organizations . where ( " organizations.id not in ( #{ project_org_ids . join ( ',' ) } ) and organizations.name like ? " , condition ) . page ( params [ :page ] . to_i || 1 ) . per ( limit )
@org_count = User . current . organizations . where ( " organizations.id not in ( #{ project_org_ids . join ( ',' ) } ) and organizations.name like ' #{ condition } ' " ) . count
# if project_org_ids.empty?
# @orgs_not_in_project = Organization.where("(is_public or creator_id =?) = 1 and name like ?",User.current.id, condition).page((params[:page].to_i || 1)).per(limit)
# @org_count = Organization.where("is_public = 1 or creator_id =?", User.current.id).where("name like ?", condition).count
# else
# project_org_ids = "(" + project_org_ids.join(',') + ")"
# @orgs_not_in_project = Organization.where("id not in #{project_org_ids} and (is_public = 1 or creator_id =?) and name like ?", User.current.id, condition).page((params[:page].to_i || 1)).per(limit)
# @org_count = Organization.where("id not in #{project_org_ids} and (is_public = 1 or creator_id =?)", User.current.id).where("name like ?", condition).count
# end
@orgs_page = Paginator . new @org_count , limit , params [ :page ]
@no_roll_hint = params [ :hint_flag ]
#render :json => {:orgs => @orgs_not_in_project, :count => @org_count}.to_json
respond_to do | format |
format . js
end
end
def index
render_404
end
def courserender_404
end
# Time 2015-01-29 11:19:11
# Author lizanle
# Description 项目搜索方法
def search
# 如果有名字,就按名字搜索,如果没有,就展示所有,用Karminari分页
if params [ :name ] . present?
@project_pages = Project . project_entities . visible . like ( params [ :name ] ) . page ( params [ :page ] ) . per ( 10 )
else
@project_pages = Project . project_entities . visible . page ( params [ :page ] ) . per ( 10 )
end
@projects = @project_pages . order ( " created_on desc " )
@limit = 10 #per_page_option
@project_count = Project . project_entities . visible . like ( params [ :name ] ) . page ( params [ :page ] ) . count
@project_pages = Paginator . new @project_count , @limit , params [ 'page' ]
@name = params [ :name ]
@type = 'projects'
respond_to do | format |
format . html {
render :layout = > 'base'
scope = Project
unless params [ :closed ]
scope = scope . active
end
}
# 需要到处atom使用的格式 ( redmine自带)
format . atom {
projects = Project . visible . order ( 'created_on DESC' ) . limit ( Setting . feeds_limit . to_i ) . all
render_feed ( projects , :title = > " #{ Setting . app_title } : #{ l ( :label_project_latest ) } " )
}
end
end
# Time 2015-01-29 16:13:20
# Author lizanle
# Description 项目首页中用户反馈 方法
def feedback
@page = params [ :page ] . to_i
# Find the page of the requested reply
@jours = @project . journals_for_messages . where ( 'm_parent_id IS NULL' ) . order ( 'created_on DESC' )
limit = 10
offset = @jours . count ( :conditions = > [ " #{ JournalsForMessage . table_name } .id > ? " , params [ :r ] . to_i ] )
page = 1 + offset / limit
if params [ :r ] && @page . nil?
@page = @page < 0 ? 1 : @page
end
@page = @page > page ? page : @page
@jour = paginateHelper @jours , 10
@state = false
respond_to do | format |
format . html
format . api
end
end
def project_respond
project_id = request . headers [ " Referer " ] . match ( ( %r|/([0-9]{1,})/| ) ) [ 1 ]
parent_id = params [ :reference_id ]
author_id = User . current . id
reply_user_id = params [ :reference_user_id ]
reply_id = params [ :reference_message_id ]
content = params [ :project_respond ]
options = { :user_id = > author_id ,
:m_parent_id = > parent_id ,
:m_reply_id = > reply_id ,
:reply_id = > reply_user_id ,
:notes = > content ,
:is_readed = > false }
@jfm = Project . add_new_jour ( nil , nil , project_id , options )
@save_succ = @jfm . errors . empty?
respond_to do | format |
format . js
end
end
def new
if User . current . login? && User . current . mail . present?
@issue_custom_fields = IssueCustomField . sorted . all
@trackers = Tracker . sorted . all
@project = Project . new
@project . safe_attributes = params [ :project ]
if params [ :course_id ]
@course = Course . find params [ :course_id ]
elsif params [ :contest_id ]
@contest = Contest . find params [ :contest_id ]
end
render :layout = > 'base_edu'
elsif User . current . mail . blank?
redirect_to my_account_path
else
redirect_to signin_url
end
end
def share
@shares = @project . shares . reverse
@base_courses_tag = @project . project_type
respond_to do | format |
format . html { render :layout = > 'base_courses' if @base_courses_tag == 1 }
format . api
end
end
# 注意: 修改该方法的时候注意同步修改forked方法
# forked方法也会创建项目
def create
unless User . current . login?
redirect_to signin_url
return
end
ActiveRecord :: Base . transaction do
begin
# 创建项目
@project = Project . new
@project . safe_attributes = params [ :project ]
@project . user_id = User . current . id
params [ :project ] [ :is_public ] ? @project . is_public = 1 : @project . is_public = 0
if @project . save!
r = Role . givable . find_by_id ( Setting . new_project_user_role_id . to_i ) || Role . givable . first
m = Member . new ( :user = > User . current , :roles = > [ r ] )
# project's score
if ProjectScore . where ( " project_id=? " , @project . id ) . first . nil?
ProjectScore . create ( :project_id = > @project . id , :score = > 0 )
end
# end
project_info = ProjectInfo . new ( :user_id = > User . current . id , :project_id = > @project . id )
@project . members << m
@project . project_infos << project_info
identifier = generate_identifier
logger . info ( " project rep_identifier is #{ identifier } " )
repository = Repository . new
repository . type = 'Repository::Gitlab'
repository . identifier = identifier
repository . url = identifier
repository . project_id = @project . id
repository . save!
s = Trustie :: Gitlab :: Sync . new
gproject = s . create_project ( @project , repository )
raise " 版本库创建失败 " if gproject . try ( :id ) . blank?
@project . update_column ( :rep_identifier , identifier )
flash [ :notice ] = l ( :notice_successful_create )
redirect_to settings_project_url ( @project )
end
rescue Exception = > e
logger . error ( " failed: create repository #{ e . message } " )
flash [ :notice ] = " 项目创建失败,请稍后重试 "
redirect_to new_project_path
raise ActiveRecord :: Rollback
end
end
end
# 随机生成唯一标识
def generate_identifier
code = " p " + DCODES . sample ( 8 ) . join
return generate_identifier if Project . where ( rep_identifier : code ) . present?
code
end
def copy
@issue_custom_fields = IssueCustomField . sorted . all
@trackers = Tracker . sorted . all
@source_project = Project . find ( params [ :id ] )
if request . get?
@project = Project . copy_from ( @source_project )
@project . identifier = Project . next_identifier if Setting . sequential_project_identifiers?
else
Mailer . with_deliveries ( params [ :notifications ] == '1' ) do
@project = Project . new
@project . safe_attributes = params [ :project ]
if validate_parent_id && @project . copy ( @source_project , :only = > params [ :only ] )
@project . set_allowed_parent! ( params [ :project ] [ 'parent_id' ] ) if params [ :project ] . has_key? ( 'parent_id' )
flash [ :notice ] = l ( :notice_successful_create )
redirect_to settings_project_url ( @project )
elsif ! @project . new_record?
# Project was created
# But some objects were not copied due to validation failures
# (eg. issues from disabled trackers)
# TODO: inform about that
redirect_to settings_project_url ( @project )
end
end
end
rescue ActiveRecord :: RecordNotFound
# source_project not found
render_404
end
# Time 2015-01-29 10:42:00
# Author lizanle
# Description 项目动态展示方法,删除了不必要的代码
def show
# 顶部导航
@project_menu_type = 1
if params [ :jump ] && redirect_to_project_menu_item ( @project , params [ :jump ] )
return
end
@author = params [ :user_id ] . blank? ? nil : User . active . find ( params [ :user_id ] )
@page = params [ :page ] ? params [ :page ] . to_i + 1 : 0
# 根据私密性,取出符合条件的所有数据
if User . current . member_of? ( @project ) || User . current . admin?
case params [ :type ]
when nil
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type in ('Issue', 'TrainingTask','Message','News', 'Project', 'Attachment','Commit','AppliedProject') " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'issue'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'Issue' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'training_task'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'TrainingTask' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'news'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'News' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'message'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'Message' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'attachment'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'Attachment' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
when 'applied_project'
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type = 'AppliedProject' " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
end
else
@events_pages = ForgeActivity . includes ( :project ) . where ( " forge_activities.project_id = ? and projects.is_public = ? and forge_act_type != ? " , @project , 1 , " Document " ) . order ( " created_at desc " ) . page ( params [ 'page' || 1 ] ) . per ( 10 ) ;
end
@type = params [ :type ]
# 版本库统计图
# unless @project.gpid.nil? || @project.project_score.changeset_num == 0
# # rep_statics_commit = @project.rep_statics.order("commits_num desc")
# rep_statics_commit = RepStatics.find_by_sql("SELECT * FROM `rep_statics` where project_id = #{@project.id} order by commits_num desc limit 10")
# rep_statics_code = RepStatics.find_by_sql("SELECT * FROM `rep_statics` where project_id = #{@project.id} order by changeset desc limit 10")
# # rep_statics_code = @project.rep_statics.sort_by {|u| u.changeset}.reverse
# @a_uname = rep_statics_commit.map {|s| s.uname }
# @a_uname_code = rep_statics_code.map {|s| s.uname }
# @a_commits_num = rep_statics_commit.map {|s| s.commits_num.to_i }
# @a_commits_add = rep_statics_code.map {|s| s.add.to_i }
# @a_commits_del = rep_statics_code.map {|s| s.del.to_i }
# @a_commits_changeset = rep_statics_code.map {|s| s.changeset.to_i }
# g = Gitlab.client
# begin
# gid = @project.gpid
# g_project = g.project(gid)
# g_branch = g_project.default_branch.to_s
# rescue =>e
# logger.error("get default branch failed: " + e)
# end
# @rev = g_branch.nil? ? "master" : g_branch
# end
# 根据对应的请求,返回对应的数据
respond_to do | format |
format . html
format . api
format . js
end
end
def settings
# 顶部导航
@project_menu_type = 10
if @project . is_child_training_project?
return render_404
end
# 修改查看消息状态
applied_messages = ForgeMessage . where ( " user_id =? and project_id =? and forge_message_type =? and viewed =? " , User . current . id , @project , " AppliedProject " , 0 )
applied_messages . update_all ( :viewed = > true )
@issue_custom_fields = IssueCustomField . sorted . all
@issue_category || = IssueCategory . new
@member || = @project . members . new
@trackers = Tracker . sorted . all
@wiki || = @project . wiki
@select_tab = params [ :tab ]
#找出所有不属于项目的公共组织
project_org_ids = OrgProject . find_by_sql ( " select distinct organization_id from org_projects where project_id = #{ @project . id } " )
if project_org_ids . empty?
@orgs_not_in_project = Organization . where ( " is_public = 1 " )
else
project_org_ids = " ( " + project_org_ids . join ( ',' ) + " ) "
@orgs_not_in_project = Organization . where ( " id not in #{ project_org_ids } and is_public = 1 " )
end
# 里程碑
@versions = @project . shared_versions . sort
# 处理从新建版本库返回来的错误信息
if ! params [ :repository_error_message ] . to_s . blank?
html = " "
errors = params [ :repository_error_message ] . flatten
errors . each do | error |
# 版本库路径为空的错误信息不予提示
if ( error! = l ( :label_repository_path_not_null ) )
html << error << " ; "
end
end
if params [ :repository ] == " pswd_is_null "
html << l ( :label_password_not_null )
end
flash . now [ :error ] = html if ! html . to_s . blank?
end
# # for: 设置默认分支
# @gitlab_repository = Repository.where(:project_id => @project, :type => "Repository::Gitlab").first
# unless @gitlab_repository.nil?
# gitlab_address = Redmine::Configuration['gitlab_address']
# creator = @project.owner.try(:login)
# @repos_url = gitlab_address+"/" + creator + "/" + @gitlab_repository.identifier+"."+"git"
# end
#
# scm = params[:repository_scm] || (Redmine::Scm::Base.all & Setting.enabled_scm).first
# @repository = Repository.factory(scm)
# @repository.is_default = @project.repository.nil?
# @repository.project = @project
# @gitlab_rep = Repository.where(:type => "Repository::Gitlab", :project_id => @project).first
# unless @project.gpid.nil?
# g = Gitlab.client
# @gitlab_branches = g.branches(@project.gpid)
# @branch_names = @gitlab_branches.map{|b| b.name}
# @gitlab_default_branch = g.project(@project.gpid).default_branch
# end
end
# 项目邀请用户加入实现过程
# 两种情况: 1、系统外用户; 2、系统内用户 (通过邮件判定)
def send_mail_to_member
# 该邮箱未注册过
if ! params [ :mail ] . blank? && User . find_by_mail ( params [ :mail ] . to_s ) . nil?
if ! User . where ( " login =? " , params [ :mail ] ) . first . nil?
# 用户名唯一,用户修改邮箱,未修改用户名,用户名等同邮箱的情况,默认改用户已经注册
user = User . find_by_login ( params [ :mail ] . to_s )
if ! user . member_of? ( @project )
# 如果已经邀请过该用户,则不重复发送
if InviteList . where ( " project_id =? and mail =? " , @project . id , params [ :mail ] . to_s ) . first . nil?
email = params [ :mail ]
# Mailer.request_member_to_project(email, @project, User.current).deliver
flash [ :notice ] = l ( :notice_email_sent , :value = > email )
else
flash [ :error ] = l ( :notice_email_invited )
end
else
flash [ :error ] = l ( :label_member_of_project , :value = > email )
end
else
email = params [ :mail ]
first_name = params [ :first_name ]
last_name = params [ :last_name ]
gender = params [ :gender ]
# Mailer.send_invite_in_project(email, @project, User.current, first_name, last_name, gender).deliver
@is_zhuce = false
flash [ :notice ] = l ( :notice_email_sent , :value = > email )
end
# 邮箱地址已被注册
elsif ! User . find_by_mail ( params [ :mail ] . to_s ) . nil?
user = User . find_by_mail ( params [ :mail ] . to_s )
if ! user . member_of? ( @project )
# 如果已经邀请过该用户,则不重复发送
invite_list = InviteList . where ( " project_id =? and mail =? " , @project . id , params [ :mail ] . to_s ) . first
if invite_list . nil?
email = params [ :mail ]
# Mailer.request_member_to_project(email, @project, User.current).deliver
flash [ :notice ] = l ( :notice_email_sent , :value = > email )
else
# 已经发送过了, 则隔3小时才能再次发送
if Time . now - invite_list . created_at > 10800
email = params [ :mail ]
# Mailer.request_member_to_project(email, @project, User.current).deliver
flash [ :notice ] = l ( :notice_email_sent , :value = > email )
else
flash [ :error ] = l ( :notice_email_invited )
end
end
else
flash [ :error ] = l ( :label_member_of_project , :value = > email )
end
else
@is_zhuce = true
end
respond_to do | format |
format . html { redirect_to invite_members_by_mail_project_url ( @project ) }
end
end
# 发送邮件邀请新用户页面对应方法
def invite_members_by_mail
if User . current . member_of? ( @project ) || User . current . admin?
@inviter_lists = InviteList . where ( project_id : @project . id ) . order ( " created_at desc " )
# @inviters = []
# @waiters = []
# unless @inviter_lists.blank?
# @inviter_lists.each do|inviter_list|
# unless inviter_list.user.nil?
# if inviter_list.user.member_of?(@project)
# @inviters << inviter_list.user
# @inviters_count = @inviters.size
# else
# @waiters << inviter_list.user
# @waiters_count = @waiters.size
# end
# end
# end
# end
@is_zhuce = false
respond_to do | format |
format . html
format . js
end
else
render_403
end
end
# 邀请Trustie注册用户
# def invite_members
# if User.current.member_of?(@project) || User.current.admin?
# @member ||= @project.members.new
# respond_to do |format|
# format.html
# end
# else
# render_403
# end
# end
def edit
end
def set_public_or_private
@project = Project . find ( params [ :id ] )
if @project . is_public?
@project . update_attribute ( :is_public , 0 )
else
@project . update_attribute ( :is_public , 1 )
end
end
def project_watcherlist
unless @project . nil?
if ! @project . is_public? && ! User . current . member_of? ( @project ) && ! User . current . admin?
render_403
else
@users -= watched . watcher_users if @watched
end
@watchers = @project . watcher_users
@limit = 32
@is_remote = true
@watchers_count = @watchers . count
@watcher_pages = Paginator . new @watchers_count , @limit , params [ 'page' ] || 1
@offset || = @watcher_pages . offset
@watchers = paginateHelper @watchers , @limit
end
end
# include CoursesHelper
def member
# 消息"同意加入项目"
@project_menu_type = 8
# if params[:message_id]
# message_invite(params[:message_id], params[:key])
# end
# update_messsages_to_viewed("ForgeMessage", @project)
# params[:login]为邮箱邀请用户加入,主要功能:
# 1、自动注册
# 2、加入项目、创建角色
# 3、用户得分
# if params[:mail]
# userid = Token.find_by_value(params[:token]).user_id
# user = User.find(userid)
# user.activate!
# Member.create(:role_ids => [4], :user_id => userid, :project_id => params[:id])
# UserGrade.create(:user_id => userid, :project_id => params[:id])
# token = Token.get_token_from_user(user, 'autologin')
# #user = User.try_to_autologin(token.value)
# if user
# start_user_session(user)
# user.save
# redirect_to project_member_path(params[:id])
# return
# # account_ project_member_path(params[:id])
# flash[:notice] = l(:label_mail_invite_success)
# end
# end
# 私有项目非项目成员无法访问成员列表
unless @project . is_public?
return render_403 unless ( User . current . member_of? ( @project ) || User . current . admin? )
end
## 有角色参数的才是课程,没有的就是项目
@render_file = 'project_member_list'
# 判断是否课程
= begin
if @project . project_type == Project :: ProjectType_course
@teachers = searchTeacherAndAssistant ( @project )
@canShowCode = isCourseTeacher ( User . current . id )
case params [ :role ]
when '1'
@subPage_title = l :label_teacher_list
@members = searchTeacherAndAssistant ( @project )
when '2'
@subPage_title = l :label_student_list
@members = searchStudent ( @project )
else
@subPage_title = ''
@members = @project . member_principals . includes ( :roles , :principal ) . all . sort
end
else
if ! @project . is_public? && ! User . current . member_of? ( @project ) && ! User . current . admin?
render_403
else
roles = Role . find_all_givable
@subPage_title = l :label_member_list
@members = @project . member_principals . includes ( :roles , :principal ) . joins ( " LEFT JOIN #{ OptionNumber . table_name } ON #{ OptionNumber . table_name } .user_id = #{ Member . table_name } .user_id and #{ OptionNumber . table_name } .score_type = 2 AND #{ Member . table_name } .project_id = #{ OptionNumber . table_name } .project_id " ) . order ( " #{ OptionNumber . table_name } .total_score DESC " ) . all
@applied_members = appied_project_members ( @project , @members )
end
end
= end
@members = @project . member_principals . reorder ( " members.id asc " ) . includes ( :roles , :principal ) . all
@members = paginateHelper @members , 20
end
# 搜索非项目成员的用户
def search_none_user
if params [ :search ] && params [ :search ] . lstrip . rstrip != " "
scope = Principal . active . sorted . not_member_of ( @project ) . where ( " mail is not null and mail != '' " ) . like ( params [ :search ] )
else
scope = Principal . active . sorted . not_member_of ( @project ) . where ( " mail is not null and mail != '' " )
end
@members = paginateHelper scope , 5
end
def member_forked
@forked_projects = Project . where ( :forked_from_project_id = > @project . id )
@limit = 32
@is_remote = true
@forked_count = @forked_projects . count
@forked_pages = Paginator . new @forked_count , @limit , params [ 'page' ] || 1
@offset || = @forked_pages . offset
@forked_projects = paginateHelper @forked_projects , @limit
# @forked_members = User.find_by_sql("SELECT u.* FROM `projects` p,`users` u where p.user_id = u.id and p.forked_from_project_id = #{@project.id} ;")
end
def update_message_status ( user , project )
# 更新加入项目消息
project__messages = ForgeMessage . where ( " forge_message_type in ('ProjectInvite', 'JoinProject', 'RemoveFromProject') and user_id =? and project_id =? " , user , project )
project__messages . update_all ( :viewed = > true ) unless project__messages . blank?
end
def message_invite ( message_id , key )
forge_message = ForgeMessage . find ( message_id )
if key == forge_message . secret_key
# 情况:用户收到邀请邮件还没看,但是管理员已经把该用户添加进项目
if Member . where ( " user_id =? and project_id =? " , forge_message . user_id , forge_message . project_id ) . first . nil?
Member . create ( :role_ids = > [ 4 ] , :user_id = > forge_message . user_id , :project_id = > forge_message . project_id )
UserGrade . create ( :user_id = > forge_message . user_id , :project_id = > forge_message . project_id )
end
end
end
#判断指定用户是否为课程教师
def isCourseTeacher ( id )
result = false
if @teachers . find_by_user_id ( id ) != nil
result = true
end
result
end
def sort_project_members project , members
#userGrade = UserGrade.where(:project_id => project.id)
users = UserGrade . where ( :project_id = > project . id ) .
order ( 'grade DESC' ) .
joins ( " LEFT JOIN users ON users.id = user_grades.id " ) .
select ( " DISTINCT user_grades.user_id " )
memberlist = [ ]
users . each do | user |
members . each do | member |
if member [ :user_id ] == user [ :user_id ]
memberlist << member
end
end
end
memberlist
end
def appied_project_members ( project , members )
users = AppliedProject . where ( :project_id = > project . id )
memberlist = [ ]
users . each do | user |
members . each do | member |
if member [ :user_id ] == user [ :user_id ]
memberlist << member
end
end
end
memberlist
end
def file
end
def statistics
end
#end
# 获取项目tree目录的最新提交记录
# 异步请求
# gpid, rev, ent_name, g
def repository_tree_changes
rev = params [ :rev ]
ent_path = params [ :ent_path ]
gpid = params [ :gpid ]
g = Gitlab . client
begin
result = g . rep_last_changes ( gpid , :rev = > rev , :path = > ent_path )
result = { :message = > result . message , :author_name = > User . find_by_mail ( result . author_email ) . nil? ? result . author_email : User . find_by_mail ( result . author_email ) . show_name , :time = > distance_of_time_in_words ( result . time , Time . now ) }
rescue Exception = > e
puts e
end
render :json = > result
end
def update
@project . safe_attributes = params [ :project ]
@project . organization_id = params [ :organization_id ]
params [ :project ] [ :is_public ] == " on " ? @project . is_public = 1 : @project . is_public = 0
params [ :project ] [ :hidden_repo ] == " on " ? @project . hidden_repo = 1 : @project . hidden_repo = 0
# # 更新公开私有时同步gitlab公开私有
# if !@project.gpid.nil? && @project.is_public != (params[:project][:is_public] == "on" ? 1 : 0)
# g = Gitlab.client
# params[:project][:is_public] == "on" ? g.edit_project(@project.gpid, 20, params[:branch]) : g.edit_project(@project.gpid, 0, params[:branch])
# end
# end
if validate_parent_id && @project . save
@project . set_allowed_parent! ( params [ :project ] [ 'parent_id' ] ) if params [ :project ] . has_key? ( 'parent_id' )
if params [ :project ] [ :is_public ] == '0'
project_status = ProjectStatus . find_by_project_id ( @project . id )
project_status . destroy if project_status
elsif params [ :project ] [ :is_public ] == '1'
project_status = ProjectStatus . create ( :project_id = > @project . id , :watchers_count = > @project . watchers . count , :changesets_count = > @project . changesets . count , :grade = > 0 , :project_type = > @project . project_type )
end
respond_to do | format |
format . html {
flash [ :notice ] = l ( :notice_successful_update )
redirect_to settings_project_url ( @project , :course = > @project . project_type )
}
format . api { render_api_ok }
end
else
respond_to do | format |
format . html {
settings
render :action = > 'settings'
}
format . api { render_validation_errors ( @project ) }
end
end
end
def add_script
if User . current . admin? || User . current . member_of? ( @project )
if @project . update_attribute ( :script , params [ :project_script ] )
@notice = " 脚本添加成功 "
else
@notice = " 脚本添加失败 "
end
else
return render_403
end
end
def modules
@project . enabled_module_names = params [ :enabled_module_names ]
flash [ :notice ] = l ( :notice_successful_update )
redirect_to settings_project_url ( @project , :tab = > 'modules' )
end
GITLABTYPE = " Repository::Gitlab "
def archive
if request . post?
if @project . destroy && @project . gpid
# 删除版本库信息
begin
g = Gitlab . client
g . delete_project ( @project . gpid )
rescue Exception = > e
puts e
end
# 删除Trustie版本库记录
repoisitory = Repository . where ( :project_id = > @project . id , :type = > GITLABTYPE ) . first
repoisitory . delete
@project . update_column ( :gpid , nil )
@project . update_column ( :forked_from_project_id , nil )
else
flash [ :error ] = l ( :error_can_not_archive_project )
end
unless @project . archive
end
end
if params [ :type ] == " project "
redirect_to user_path ( User . current )
else
redirect_to admin_projects_url ( :status = > params [ :status ] )
end
end
def unarchive
@project . unarchive if request . post? && ! @project . active?
redirect_to admin_projects_url ( :status = > params [ :status ] )
end
# 弹框提醒:
# 自己不能参加自己的实训项目
# 没有建立版本库的项目不能开启实训
# 已经实训过直接跳入
#
def training_chiled_project_exec
if ! User . current . logged?
return render_403
end
respond_to do | format |
format . js
end
end
# training_status: 默认为0; 1代表实训项目; 2: 代表实训子项目
def training_project_execute
if @project . training_tasks . count == 0
@notice = " 实训开启失败:请先发布实训任务 "
return
elsif Repository . where ( :project_id = > @project . id , :type = > " Repository::Gitlab " ) . count == 0
@notice = " 实训开启失败:请先创建版本库 "
return
end
jobName = " #{ @project . id } "
pipeLine = " #{ Base64 . encode64 ( @project . script ) } "
uri = URI ( " http://123.59.135.74:9999/jenkins-exec/api/createJob " )
params = { jobName : jobName , pipeLine : pipeLine }
res = uri_exec uri , params
training_project_notice res
if res [ 'code' ] == 0
@project . update_attribute ( :training_status , 1 )
end
end
def training_project_update
jobName = " #{ @project . id } "
pipeLine = " #{ Base64 . encode64 ( @project . script ) } "
uri = URI ( " http://123.59.135.74:9999/jenkins-exec/api/updateJob " )
params = { jobName : jobName , pipeLine : pipeLine }
res = uri_exec uri , params
training_project_notice res
if res [ 'code' ] == 0
@project . update_attribute ( :training_status , 1 )
end
end
def training_project_notice res
if res [ 'code' ] == 0
@notice = " 实训开启成功 "
elsif res [ 'code' ] == - 2
@notice = " Job已存在 "
else
@notice = res [ 'msg' ] . nil? ? " 实训开启失败 " : res [ 'msg' ]
end
end
# TrainintTask.status 0: 评测中 1: 评测成功 2: 评测进行中
# 非项目成员、非实训子项目不允许执行该方法
def task_execute
if ! @project . is_child_training_project? || ! User . current . member_of? ( @project )
return render_403
end
taskId = params [ :training_task_id ]
jobName = @project . forked_from_project_id
@training_task = TrainingTask . find ( taskId )
step = @training_task . position
rep_identify = Repository . where ( :project_id = > @project . id , :type = > " Repository::Gitlab " ) . first . try ( :identifier )
gitlab_address = Redmine :: Configuration [ 'gitlab_address' ]
gitUrl = gitlab_address . to_s + " / " + @project . owner . to_s + " / " + rep_identify + " . " + " git "
gitUrl = Base64 . encode64 ( gitUrl )
if @training_task . status == 0
params = { :jobName = > " #{ jobName } " , :taskId = > " #{ taskId } " , :step = > " #{ step } " , :gitUrl = > " #{ gitUrl } " }
uri = URI . parse ( " http://123.59.135.74:9999/jenkins-exec/api/buildJob " )
begin
res = uri_exec uri , params
# 轮询获取
for i in 0 .. 60 do
sleep ( 1 )
result = TrainingTask . where ( :id = > @training_task . id ) . first . try ( :result )
if ( result != 0 )
break
end
end
# @training_task = TrainingTask.find(@training_task.id)
@page = params [ :page ] ? params [ :page ] . to_i + 1 : 0
@events_pages = ForgeActivity . where ( " project_id = ? and forge_act_type in ('Issue', 'TrainingTask','Message','News', 'Project', 'Attachment','Commit') " , @project ) . includes ( :forge_act ) . order ( " updated_at desc " ) . limit ( 10 ) . offset ( @page * 10 )
rescue Exception = > e
puts e
end
end
respond_to do | format |
format . js
end
end
# def uri_exec uri, params
# # res = Net::HTTP.post_form(uri, params).body
# # res = JSON.parse(res)
# end
# 开启实训项目,学生会fork一个项目并自动发送任务
def training_project_extend
@project = Project . find ( params [ :id ] )
@repository = Repository . where ( " project_id =? and type =? " , @project . id , " Repository::Gitlab " )
# 如果当前用户已经fork过该项目, 不会新fork项目, 则跳至已fork的项
unless has_forked? ( @project , User . current )
project = project_from_current_project ( @project . id , User . current . id )
redirect_to project_path ( project )
else
ActiveRecord :: Base . transaction do
g = Gitlab . client
if User . current . gid . nil?
s = Trustie :: Gitlab :: Sync . new
s . sync_user ( User . current )
end
gproject = g . fork ( @project . gpid , User . current . gid )
if gproject
new_training_project = copy_project_and_module ( @project , gproject )
forked_count = @project . forked_count . to_i + 1
@project . update_attributes ( :forked_count = > forked_count )
# 发布实训任务,只发布实训任务的第一个
publish_training_tasks ( @project . id , new_training_project . id , 1 , @project . user_id )
end
end
end
end
# 需要传Task ID
# 判断任务是否完成
# 如果完成则发送下一个任务直到任务结束
# TrainingTask.status 1 成功,其它失败
def training_task_status
status = params [ :status ] . to_i
task_id = params [ :taskId ]
message = Base64 . decode64 ( params [ :msg ] ) unless params [ :msg ] . blank?
begin
@training_task = TrainingTask . find ( task_id )
# 如果已经执行成功过,则不重复执行
return if @training_task . status == 1
original_project_id = Project . find ( @training_task . project_id ) . try ( :forked_from_project_id )
original_project = Project . find ( original_project_id )
rescue
return
end
original_tasks_count = original_project . training_tasks . count
position = @training_task . try ( :position )
# 测试,默认成功
if status == 0
ActiveRecord :: Base . transaction do
if position < original_tasks_count
# 继续发布下一个任务
publish_training_tasks original_project_id , @training_task . project_id , position + 1 , original_project . user_id
end
@training_task . update_attribute ( :status , 1 )
@training_task . update_attribute ( :result , 1 )
# 创建一条回复提醒
content = ( position == original_tasks_count ) ? " 恭喜您,您已经完成了实训项目的所有任务 " : " 恭喜您,您已经完成了本任务,请继续下一任务 "
add_training_task_journal ( content , original_project . user_id )
end
else
content = " 很抱歉,您的任务未通过,请继续加油,错误信息如下: #{ message } "
add_training_task_journal ( content , original_project . user_id )
# 失败的时候可以继续提交
@training_task . update_attribute ( :status , 0 )
@training_task . update_attribute ( :result , 2 )
end
end
# 创建一条回复
def add_training_task_journal content , user_id
jour = Journal . new
jour . user_id = user_id
jour . notes = content
jour . journalized = @training_task
jour . save
update_user_activity ( @training_task . class , @training_task . id )
update_forge_activity ( @training_task . class , @training_task . id )
end
# 实训开启成功后,发布第一个任务
# REDO:失败后提醒用户,及相关处理
def publish_training_tasks original_project_id , new_training_project_id , position , user_id
original_task = TrainingTask . where ( :project_id = > original_project_id , :position = > position ) . first
training_task = TrainingTask . new
training_task . save_attachments ( params [ :attachments ] || ( params [ :training_task ] && params [ :training_task ] [ :uploads ] ) )
training_task . subject = original_task . subject
training_task . description = original_task . description
training_task . position = original_task . position
training_task . project_id = new_training_project_id
training_task . author_id = User . current . id
begin
training_task . save
rescue Exception = > e
puts e
end
end
# 复制项目
def copy_project_and_module tproject , gproject
project = Project . new
project . name = tproject . name
project . is_public = tproject . is_public
project . status = tproject . status
project . description = tproject . description
project . hidden_repo = tproject . hidden_repo
project . user_id = User . current . id
project . project_type = 0
project . project_new_type = tproject . project_new_type
project . gpid = gproject . id
project . forked_from_project_id = tproject . id
project . enabled_module_names = tproject . enabled_module_names
if project . save
project . update_attribute ( :training_status , 2 )
r = Role . givable . find_by_id ( Setting . new_project_user_role_id . to_i ) || Role . givable . first
m = Member . new ( :user = > User . current , :roles = > [ r ] )
if ProjectScore . where ( " project_id=? " , project . id ) . first . nil?
ProjectScore . create ( :project_id = > project . id , :score = > false )
end
project_info = ProjectInfo . new ( :user_id = > User . current . id , :project_id = > project . id )
user_grades = UserGrade . create ( :user_id = > User . current . id , :project_id = > project . id )
Rails . logger . debug " UserGrade created: #{ user_grades . to_json } "
project_status = ProjectStatus . create ( :project_id = > @project . id , :watchers_count = > 0 , :changesets_count = > 0 , :project_type = > @project . project_type , :grade = > 0 )
Rails . logger . debug " ProjectStatus created: #{ project_status . to_json } "
project . members << m
project . project_infos << project_info
copy_repository ( project , gproject )
return project
else
respond_to do | format |
format . html { render :action = > 'forked' , :layout = > 'base_projects' }
format . api { render_validation_errors ( @project ) }
end
end
end
# 判断用户是否已经fork过该项目
def has_forked? ( project , user )
projects = Project . where ( " user_id =? " , user )
projects . map ( & :forked_from_project_id ) . detect { | s | s == @project . id } . nil? ? true : false
end
def copy_repository ( project , gproject )
repository = Repository . factory ( 'Git' )
repository . project_id = project . id
repository . type = 'Repository::Gitlab'
repository . url = gproject . name
repository . identifier = gproject . name
repository = repository . save
end
# 资源库fork弹框
def forked_pop
respond_to do | format |
format . js
end
end
# 配置成员弹框
def delete_member_pop
@member = Member . find ( params [ :member ] . to_i )
respond_to do | format |
format . js
end
end
def close
@project . close
redirect_to project_url ( @project )
end
def reopen
@project . reopen
redirect_to project_url ( @project )
end
REP_TYPE = " Repository::Gitlab "
# Delete @project
def destroy
ActiveRecord :: Base . transaction do
g = Gitlab . client
g . delete_project ( @project . gpid )
# 删除Trustie版本库记录
repoisitory = Repository . where ( :project_id = > @project . id , :type = > GITLABTYPE ) . first
repoisitory . delete
@project . update_column ( :gpid , nil )
@project . update_column ( :forked_from_project_id , nil )
Tiding . where ( :container_id = > @project . id , :container_type = > [ " JoinProject " , " DealProject " , " ReporterJoinProject " , " ManagerJoinProject " ] ) . destroy_all
@project_to_destroy = @project
@project_to_destroy . destroy
end
respond_to do | format |
format . js
if params [ :type ] == " project "
format . html { redirect_to user_path ( User . current ) }
else
format . html { redirect_to admin_projects_url ( :status = > params [ :status ] ) }
end
end
# hide project in layout
@project = nil
end
# Delete @project's repository
def destroy_repository
if is_project_manager? ( User . current . id , @project . id )
@gitlab_repository = Repository . where ( :project_id = > @project , :type = > REP_TYPE ) . first
@is_true = params [ :is_true ]
unless @is_true . nil?
begin
g = Gitlab . client
d_project = g . delete_project ( @project . gpid )
if d_project
@gitlab_repository . destroy
@project . update_attribute ( :gpid , nil )
@gitlab_repository = nil
end
rescue Exception = > e
if @gitlab_repository
@gitlab_repository . destroy
@project . update_attribute ( :gpid , nil )
@gitlab_repository = nil
end
end
redirect_to new_project_repository_path ( @project )
end
else
return render_403
end
end
def show_projects_score
respond_to do | format |
format . html { render :layout = > " base_projects " }
format . js
end
end
def issue_score_index
respond_to do | format |
format . js
end
end
def news_score_index
end
def file_score_index
end
def code_submit_score_index
end
def projects_topic_score_index
end
# end
before_filter :toggleCourse , only : [ :finishcourse , :restartcourse ]
# 最好通过用户与项目的权限解决这种事情。还没写
def finishcourse
yesterday = Date . today . prev_day . to_time
@course_prefs . endup_time = yesterday
@save_flag = @course_prefs . save
respond_to do | format |
format . js
end
end
def restartcourse
day = Time . parse ( " 3000-01-01 " )
@course_prefs . endup_time = day
@save_flag = @course_prefs . save
respond_to do | format |
format . js {
render action : 'finishcourse'
}
end
end
def exit_project
@project = Project . find params [ :id ]
if User . current . login?
members = Member . where ( :user_id = > User . current . id , :project_id = > params [ :id ] ) . first
if User . current != @project . owner
members . destroy
# 移出的时候删除申请消息,不需要删除消息,所以不必要关联删除
applied_projects = AppliedProject . where ( :project_id = > @project . id , :user_id = > members . user_id ) . first
unless applied_projects . nil?
applied_projects . delete
end
end
respond_to do | format |
format . js
end
end
end
def store_mine
member = Member . where ( :project_id = > params [ :id ] , :user_id = > User . current . id ) . first
member . try ( :is_collect ) == 1 ? member . update_column ( :is_collect , 0 ) : member . update_column ( :is_collect , 1 )
end
# 项目收藏
def enshrine
@stores = Member . where ( :project_id = > params [ :id ] , :is_collect = > 1 ) . includes ( :user )
end
#加入私有项目
def join_project
respond_to do | format |
format . js
end
end
#朋友圈、科研组、开发组之间的切换
def change_project_type
@project . project_new_type = params [ :project_type ]
if @project . save
message = @project . project_new_type
else
message = " 0 "
end
render :json = > message
end
private
def memberAccess
# 如果是私有项目,项目成员不对外公开,公开项目成员列表对外公开。
unless @project . is_public?
render_403 unless User . current . member_of? ( @project )
end
end
def toggleCourse
@course_prefs = Course . find_by_extra ( @project . identifier )
unless ( @course_prefs . teacher == User . current || User . current . admin? )
render_403
end
end
def select_project_layout
project = Project . find_by_id ( params [ :id ] )
project || = @project || = @course || = params [ :course ] || = params [ :project_type ]
( project . try ( :project_type ) == Project :: ProjectType_project ) ? 'base_projects' : 'base_courses'
end
# Validates parent_id param according to user's permissions
# TODO: move it to Project model in a validation that depends on User.current
def validate_parent_id
return true if User . current . admin?
parent_id = params [ :project ] && params [ :project ] [ :parent_id ]
if parent_id || @project . new_record?
parent = parent_id . blank? ? nil : Project . find_by_id ( parent_id . to_i )
unless @project . allowed_parents . include? ( parent )
@project . errors . add :parent_id , :invalid
return false
end
end
true
end
#gcm
def desc_sort_course_by_avtivity ( activity_count , projects )
return projects if activity_count . size < 2
( activity_count . size - 2 ) . downto ( 0 ) do | i |
( 0 .. i ) . each do | j |
if activity_count [ j ] < activity_count [ j + 1 ]
projects [ j ] , projects [ j + 1 ] = projects [ j + 1 ] , projects [ j ]
activity_count [ j ] , activity_count [ j + 1 ] = activity_count [ j + 1 ] , activity_count [ j ]
end
end
end
return projects
end
def require_admin_or_manager
return unless require_login
if ! ( User . current . admin? || User . current . manager_of_project? ( @project . id ) )
render_403
return false
end
true
end
end