|
- # == Schema Information
- #
- # Table name: projects
- #
- # id :integer not null, primary key
- # name :string(190)
- # description :text(4294967295)
- # homepage :string(255) default("")
- # is_public :boolean default("1"), not null
- # parent_id :integer
- # created_on :datetime
- # updated_on :datetime
- # identifier :string(255)
- # status :integer default("1"), not null
- # lft :integer
- # rgt :integer
- # inherit_members :boolean default("0"), not null
- # project_type :integer default("0")
- # hidden_repo :boolean default("0"), not null
- # attachmenttype :integer default("1")
- # user_id :integer
- # dts_test :integer default("0")
- # enterprise_name :string(255)
- # organization_id :integer
- # project_new_type :integer
- # gpid :integer
- # forked_from_project_id :integer
- # forked_count :integer default("0")
- # publish_resource :integer default("0")
- # visits :integer default("0")
- # hot :integer default("0")
- # invite_code :string(255)
- # qrcode :string(255)
- # qrcode_expiretime :integer default("0")
- # script :text(65535)
- # training_status :integer default("0")
- # rep_identifier :string(255)
- # project_category_id :integer
- # project_language_id :integer
- # license_id :integer
- # ignore_id :integer
- # praises_count :integer default("0")
- # watchers_count :integer default("0")
- # issues_count :integer default("0")
- # pull_requests_count :integer default("0")
- # language :string(255)
- # versions_count :integer default("0")
- # issue_tags_count :integer default("0")
- # closed_issues_count :integer default("0")
- # open_devops :boolean default("0")
- # gitea_webhook_id :integer
- # open_devops_count :integer default("0")
- # recommend :boolean default("0")
- # platform :integer default("0")
- # default_branch :string(255) default("master")
- # website :string(255)
- # lesson_url :string(255)
- # use_blockchain :boolean default("0")
- # is_pinned :boolean default("0")
- # recommend_index :integer default("0")
- # pr_view_admin :boolean default("0")
- # has_actions :boolean default("0")
- #
- # Indexes
- #
- # index_projects_on_forked_from_project_id (forked_from_project_id)
- # index_projects_on_identifier (identifier)
- # index_projects_on_invite_code (invite_code)
- # index_projects_on_is_pinned (is_pinned)
- # index_projects_on_is_public (is_public)
- # index_projects_on_lft (lft)
- # index_projects_on_license_id (license_id)
- # index_projects_on_name (name)
- # index_projects_on_platform (platform)
- # index_projects_on_project_category_id (project_category_id)
- # index_projects_on_project_language_id (project_language_id)
- # index_projects_on_project_type (project_type)
- # index_projects_on_recommend (recommend)
- # index_projects_on_rgt (rgt)
- # index_projects_on_status (status)
- # index_projects_on_updated_on (updated_on)
- #
-
- class Project < ApplicationRecord
- include Matchable
- include Publicable
- include Watchable
- include ProjectOperable
- include Dcodes
-
- default_scope {where.not(id: 0)}
-
- # common:开源托管项目
- # mirror:普通镜像项目,没有定时同步功能
- # sync_mirror:同步镜像项目,有系统定时同步功能,且用户可手动同步操作
- #
- enum project_type: { sync_mirror: 2, mirror: 1, common: 0 }
-
- # forge: trustie平台项目, educoder: educoder平台项目, 默认为forge平台
- enum platform: { forge: 0, educoder: 1 }
-
- belongs_to :ignore, optional: true
- belongs_to :license, optional: true
- belongs_to :owner, class_name: 'Owner', foreign_key: :user_id, optional: true
- belongs_to :organization_extension, foreign_key: :user_id, primary_key: :organization_id, optional: true, counter_cache: :num_projects
- belongs_to :project_category, optional: true , :counter_cache => true
- belongs_to :project_language, optional: true , :counter_cache => true
- belongs_to :forked_from_project, class_name: 'Project', optional: true, foreign_key: :forked_from_project_id
- has_many :project_trends, dependent: :destroy
- has_many :watchers, as: :watchable, dependent: :destroy
- has_many :fork_users, dependent: :destroy
- has_many :forked_users, class_name: 'ForkUser', foreign_key: :fork_project_id, dependent: :destroy
- has_many :forked_projects, class_name: 'Project', foreign_key: :forked_from_project_id
- has_one :project_educoder, dependent: :destroy
-
- has_one :project_score, dependent: :destroy
- has_one :repository, dependent: :destroy
- has_many :pull_requests, dependent: :destroy
- has_many :issue_tags, -> { order("issue_tags.created_at DESC") }, dependent: :destroy
- has_many :issues, dependent: :destroy
- # has_many :user_grades, dependent: :destroy
- has_many :attachments, as: :container, dependent: :destroy
- has_one :page, dependent: :destroy
- has_one :project_score, dependent: :destroy
- has_many :versions, -> { order("versions.created_on DESC, versions.name DESC") }, dependent: :destroy
- has_many :praise_treads, as: :praise_tread_object, dependent: :destroy
- has_and_belongs_to_many :trackers, :order => "#{Tracker.table_name}.position"
- has_one :project_detail, dependent: :destroy
- has_many :project_units, dependent: :destroy
- has_one :applied_transfer_project,-> { order created_at: :desc }, dependent: :destroy
- has_many :pinned_projects, dependent: :destroy
- has_many :has_pinned_users, through: :pinned_projects, source: :owner
- has_many :webhooks, class_name: "Gitea::Webhook", primary_key: :gpid, foreign_key: :repo_id
- has_many :user_trace_tasks, dependent: :destroy
- has_many :project_invite_links, dependent: :destroy
- has_many :project_topic_ralates, dependent: :destroy
- has_many :project_topics, through: :project_topic_ralates
- has_many :commit_logs, dependent: :destroy
- has_many :daily_project_statistics, dependent: :destroy
- has_one :project_dataset, dependent: :destroy
- has_many :sync_repositories, dependent: :destroy
- has_many :home_top_settings, as: :top, dependent: :destroy
- after_create :incre_user_statistic, :incre_platform_statistic
- after_save :check_project_members
- before_save :set_invite_code, :reset_unmember_followed, :set_recommend_and_is_pinned, :reset_cache_data
- before_destroy :decre_project_common, :decre_forked_from_project_count
- after_destroy :decre_user_statistic, :decre_platform_statistic
- scope :project_statics_select, -> {select(:id,:name, :is_public, :identifier, :status, :project_type, :user_id, :forked_count, :description, :visits, :project_category_id, :project_language_id, :license_id, :ignore_id, :watchers_count, :created_on)}
- scope :no_anomory_projects, -> {where("projects.user_id is not null and projects.user_id != ?", 2)}
- scope :recommend, -> { visible.project_statics_select.where(recommend: true) }
- scope :pinned, -> {where(is_pinned: true)}
-
- delegate :content, to: :project_detail, allow_nil: true
- delegate :name, to: :license, prefix: true, allow_nil: true
-
- validate :validate_sensitive_string, on: [:create, :update]
-
- def self.all_visible(user_id=nil)
- user_projects_sql = Project.joins(:owner).where(users: {type: 'User'}).to_sql
- org_public_projects_sql = Project.joins(:owner).merge(Organization.joins(:organization_extension).where(organization_extensions: {visibility: 'common'})).to_sql
- if user_id.present?
- org_limit_projects_sql = Project.joins(:owner).merge(Organization.joins(:organization_extension).where(organization_extensions: {visibility: 'limited'})).to_sql
- org_privacy_projects_sql = Project.joins(:owner).merge(Organization.joins(:organization_extension, :organization_users).where(organization_extensions: {visibility: 'privacy'}, organization_users: {user_id: user_id})).to_sql
- return Project.from("( #{ user_projects_sql } UNION #{ org_public_projects_sql } UNION #{ org_limit_projects_sql } UNION #{org_privacy_projects_sql} ) AS projects").visible
- else
- return Project.from("( #{ user_projects_sql } UNION #{ org_public_projects_sql } ) AS projects").visible
- end
- end
-
- def reset_cache_data
- CacheAsyncResetJob.set(wait: 5.seconds).perform_later("project_common_service", self.id)
- if changes[:user_id].present?
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_count: -1}, changes[:user_id].first)
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_count: 1}, changes[:user_id].last)
- end
- if changes[:project_language_id].present?
- first_language = ProjectLanguage.find_by_id(changes[:project_language_id].first)
- last_language = ProjectLanguage.find_by_id(changes[:project_language_id].last)
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_language_count_key: first_language&.name, project_language_count: -1}, self.user_id)
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_language_count_key: last_language&.name, project_language_count: 1}, self.user_id)
- CacheAsyncSetJob.perform_later("platform_statistic_service", {project_language_count_key: first_language&.name, project_language_count: -1})
- CacheAsyncSetJob.perform_later("platform_statistic_service", {project_language_count_key: last_language&.name, project_language_count: 1})
- end
- if changes[:is_public].present?
- if changes[:is_public][0] && !changes[:is_public][1]
- CacheAsyncClearJob.set(wait: 5.seconds).perform_later('project_rank_service', self.id)
- end
- if !changes[:is_public][0] && changes[:is_public][1]
- $redis_cache.srem("v2-project-rank-deleted", self.id)
- end
- end
- if !self.common?
- CacheAsyncClearJob.set(wait: 5.seconds).perform_later('project_rank_service', self.id)
- end
- end
-
- def decre_project_common
- CacheAsyncClearJob.perform_later('project_common_service', self.id)
- end
-
- def decre_forked_from_project_count
- forked_project = self.forked_from_project
- if forked_project.present?
- forked_project.decrement(:forked_count, 1)
- forked_project.update_attribute(:forked_count, forked_project.forked_count)
- end
- end
-
- def incre_user_statistic
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_count: 1, project_language_count_key: self.project_language&.name, project_language_count: 1}, self.user_id)
- end
-
- def decre_user_statistic
- CacheAsyncSetJob.perform_later("user_statistic_service", {project_count: -1, project_language_count_key: self.project_language&.name, project_language_count: -1}, self.user_id)
- end
-
- def incre_platform_statistic
- CacheAsyncSetJob.perform_later("platform_statistic_service", {project_count: 1, project_language_count_key: self.project_language&.name, project_language_count: 1})
- end
-
- def decre_platform_statistic
- CacheAsyncSetJob.perform_later("platform_statistic_service", {project_count: -1, project_language_count_key: self.project_language&.name, project_language_count: -1})
- end
-
- def is_full_public
- owner = self.owner
- if owner.is_a?(Organization)
- return self.is_public && owner&.visibility == "common"
- else
- return self.is_public
- end
- end
-
- def reset_unmember_followed
- if changes[:is_public].present? && changes[:is_public] == [true, false]
- self.watchers.where.not(user_id: self.all_collaborators).destroy_all
- end
- end
-
- def set_invite_code
- if self.invite_code.nil?
- self.invite_code= self.generate_dcode('invite_code', 6)
- end
- end
-
- def set_recommend_and_is_pinned
- self.recommend = self.recommend_index.zero? ? false : true
- # 私有项目不允许设置精选和推荐
- unless self.is_public
- self.recommend = false
- self.recommend_index = 0
- self.is_pinned = false
- end
- end
-
- def self.search_project(search)
- # 表情处理
- search = search.to_s.each_char.select { |c| c.bytes.first < 240 }.join('')
- ransack(name_or_identifier_cont: search)
- end
- # 创建者
- def creator
- User.find(user_id).full_name
- end
-
- def members_user_infos
- members.joins(:roles).where("roles.name in ('Manager', 'Developer', 'Reporter')").joins("left join users on members.user_id = users.id ").includes(:user).where("users.type = ?", "User")
- # members.joins("left join users on members.user_id = users.id").select("users.id", "users.login","users.firstname","users.lastname")
- # .pluck("users.id", "users.login","users.lastname", "users.firstname")
- end
-
- def to_param
- self.identifier.parameterize
- end
-
- def get_issues_count(status_id)
- if status_id.present?
- self&.issues.issue_issue.select(:id, :status_id).where(status_id: status_id)&.pluck(:id).size
- else
- self&.issues.issue_issue.select(:id)&.pluck(:id).size
- end
- end
-
- def get_pull_requests_count(status_id)
- if status_id.present?
- self&.pull_requests.select(:id, :status).where(status: status_id)&.pluck(:id).size
- else
- self&.pull_requests.select(:id)&.pluck(:id).size
- end
- end
-
- #创建项目管理员
- def check_project_members
- return if owner.is_a?(Organization)
- unless members.present? && members.exists?(user_id: self.user_id)
- member_params = {
- user_id: self.user_id,
- project_id: self.id
- }
- user_member = Member.new(member_params)
- if user_member.save
- role_id = Role.select(:id,:position).where(position: 3)&.first&.id
- MemberRole.create!(member_id: user_member.id ,role_id: role_id)
- end
- end
- end
-
-
- def self.init_bluck_repository
- Project.includes(:repository).find_each do |project|
- puts project.id
- next if project.owner.blank?
- if project.repository.blank?
- puts "########### start create repositoy #############"
- Repository.create!(project_id: project.id, identifier: Project.generate_identifier, user_id: project&.owner&.id)
- end
- end
- end
-
- def self.generate_identifier
- str_arr = (("a".."z").to_a + ("A".."Z").to_a)
-
- str = str_arr.shuffle[0..8].join
- while Repository.exists?(identifier: str)
- str = str_arr.shuffle[0..8].join
- end
- str
- end
-
- def self.list_user_projects(user_id)
- projects = Project.is_private.select(:id,:user_id)
- user_not_show_1 = projects.where("user_id != ?",user_id).pluck(:id).uniq
-
- user_show_2 = projects.joins(:members).where("members.user_id = ?", user_id).pluck(:id).uniq
- Project.where.not(id: (user_not_show_1 - user_show_2).uniq)
- end
-
- def members_count
- members.select(:id).size
- end
-
-
- def can_visited?
- is_public? || User.current.admin? || member?(User.current)
- end
-
- def releases_size(current_user_id, type)
- if current_user_id == self.user_id && type.to_s == "all"
- self.repository.version_releases_count
- else
- self.repository.version_releases.releases_size
- end
- end
-
- def contributor_users
- self.pull_requests.select(:user_id).pluck(:user_id).uniq.size
- end
-
- def open_issues_count
- issues_count - closed_issues_count
- end
-
- def numerical_for_project_type
- self.class.name.constantize.project_types["#{self.project_type}"]
- end
-
- def watched_by? user
- watchers.pluck(:user_id).include? user&.id
- end
-
- def praised_by? user
- praise_treads.pluck(:user_id).include? user&.id
- end
-
- def get_premission user
- return "Owner" if owner?(user)
- return "Manager" if manager?(user)
- return "Developer" if develper?(user)
- return "Reporter" if reporter?(user)
-
- return ""
- end
-
- def fork_project
- Project.find_by(id: self.forked_from_project_id)
- end
-
- def self.members_projects(member_user_id)
- joins(:members).where(members: { user_id: member_user_id})
- end
-
- def self.find_with_namespace(namespace_path, identifier)
- logger.info "########namespace_path: #{namespace_path} ########identifier: #{identifier} "
-
- user = Owner.find_by_login namespace_path
- user = User.new(login: namespace_path) if user.nil?
- if identifier.end_with?('.json')
- project = user&.projects&.find_by(identifier: identifier) || Project.find_by(identifier: "#{namespace_path}/#{identifier}")
- identifier = identifier.sub(/.*\K.json/, '')
- project = user&.projects&.find_by(identifier: identifier) || Project.find_by(identifier: "#{namespace_path}/#{identifier}")
- else
- project = user&.projects&.find_by(identifier: identifier) || Project.find_by(identifier: "#{namespace_path}/#{identifier}")
- end
- return nil if project.blank?
-
- [project, user]
- end
-
- def ci_reactivate?
- open_devops_count > 0
- end
-
- def ci_reactivate!(ci_repo)
- ci_repo.update_column(:repo_active, 1)
- update_column(:open_devops, true)
- increment!(:open_devops_count)
- end
-
- def self.sync_educoder_shixun(url, private_token, page, per_page)
- SyncEducoderShixunJob.perform_later(url, private_token, page, per_page)
- end
-
- def self.update_common_projects_count!
- ps = ProjectStatistic.first
- ps.increment!(:common_projects_count) unless ps.blank?
- end
-
- def self.update_mirror_projects_count!
- ps = ProjectStatistic.first
- ps.increment!(:mirror_projects_count) unless ps.blank?
- end
-
- def set_updated_on(time)
- return if time.blank?
- update_column(:updated_on, time)
- end
-
- def is_transfering
- applied_transfer_project&.common? ? true : false
- end
-
- def validate_sensitive_string
- raise("项目名称包含敏感词汇,请重新输入") if name && !HarmoniousDictionary.clean?(name)
- raise("项目描述包含敏感词汇,请重新输入") if description && !HarmoniousDictionary.clean?(description)
- end
-
- def get_last_project_issues_index
- last_issue = self.issues.issue_issue.last
- deleted_issue_count = ($redis_cache.hget("issue_cache_delete_count", self.id) || 0).to_i
-
- last_issue&.project_issues_index.present? ? last_issue.project_issues_index + deleted_issue_count : 0
- end
-
- def incre_project_issue_cache_delete_count(count=1)
- $redis_cache.hincrby("issue_cache_delete_count", self.id, count)
- end
-
- def del_project_issue_cache_delete_count
- $redis_cache.hdel("issue_cache_delete_count", self.id)
- end
-
- def open_portrait
- EduSetting.get("open_portrait_projects").present? ? EduSetting.get("open_portrait_projects").split(",").include?(self.id.to_s) : false
- end
-
- def has_pull_request(branch_name)
- return true if self.pull_requests.opening.where(head: branch_name).present? || self.pull_requests.opening.where(base: branch_name).present?
- if self.forked_from_project_id.present?
- return true if self.fork_project.pull_requests.opening.where(head: branch_name).present? || self.fork_project.pull_requests.opening.where(base: branch_name).present?
- end
-
- return false
- end
-
- def self.mindspore_contributors
- cache_result = $redis_cache.get("ProjectMindsporeContributors")
- if cache_result.nil?
- contributors = []
- file = File.open('public/mindspore_authors', 'r')
- file.each_line do |l|
- itemArray = l.gsub("\r\n", "").split("|:|")
- email = itemArray[0]
- username = itemArray[1]
- commits = itemArray[2].to_i
- user = User.find_by(login: username, mail: email)
- user = User.find_by(login: username) if user.nil?
- user = User.find_by(mail: email) if user.nil?
- # next if user.nil?
- search_contributor = contributors.select{|con| con["id"]==user.id}[0]
- if search_contributor.present?
- search_contributor["contributions"] += commits
- else
- contributors << {contributions: commits, name: username, login: username, email: email, id: user&.id}.stringify_keys
- end
- end
- file.close
-
- $redis_cache.set("ProjectMindsporeContributors", contributors.to_json)
-
- return contributors
- else
- return JSON.parse(cache_result)
- end
- end
-
- def full_url
- Rails.application.config_for(:configuration)['platform_url'].to_s + '/' + self.owner&.try(:login).to_s + '/' + self.identifier.to_s
- end
-
- def to_builder
- Jbuilder.new do |project|
- project.id self.id
- project.identifier self.identifier
- project.name self.name
- project.description Nokogiri::HTML(self.description).text
- project.visits self.visits
- project.praises_count self.praises_count.to_i
- project.watchers_count self.watchers_count.to_i
- project.issues_count self.issues_count.to_i
- project.pull_requests_count self.pull_requests_count.to_i
- project.forked_count self.forked_count.to_i
- project.is_public self.is_public
- project.mirror_url self.repository&.mirror_url
- project.type self&.project_type
- project.created_at self.created_on.strftime("%Y-%m-%d %H:%M")
- project.updated_at self.updated_on.strftime("%Y-%m-%d %H:%M")
- project.forked_from_project_id self.forked_from_project_id
- project.platform self.platform
- project.author do
- if self.educoder?
- project_educoder = self.project_educoder
- project.name project_educoder&.owner
- project.type 'Educoder'
- project.login project_educoder&.repo_name.split('/')[0]
- project.image_url render_educoder_avatar_url(self.project_educoder)
- else
- user = self.owner
- project.name user.try(:show_real_name)
- project.type user&.type
- project.login user.login
- project.image_url user.get_letter_avatar_url
- end
- end
-
- project.category do
- if self.project_category.blank?
- project.nil!
- else
- project.id self.project_category.id
- project.name self.project_category.name
- end
- end
- project.language do
- if self.project_language.blank?
- project.nil!
- else
- project.id self.project_language.id
- project.name self.project_language.name
- end
- end
-
- end
- end
- end
|