diff options
| author | Kamil Trzciński <ayufan@ayufan.eu> | 2018-02-28 20:36:55 +0100 | 
|---|---|---|
| committer | Kamil Trzciński <ayufan@ayufan.eu> | 2018-02-28 20:36:55 +0100 | 
| commit | 965dc28691e2d70b7040e28d90ccbc3721a9e416 (patch) | |
| tree | 84258f35b72f2e7ce6a7198db66032df4ad5aadb /app/models/project.rb | |
| parent | e3fafa7632e038927085cf8c8228c93be44b36bd (diff) | |
| parent | 7fabc892f251740dbd9a4755baede662e6854870 (diff) | |
| download | gitlab-ce-965dc28691e2d70b7040e28d90ccbc3721a9e416.tar.gz | |
Merge commit '7fabc892f251740dbd9a4755baede662e6854870' into object-storage-ee-to-ce-backport
Diffstat (limited to 'app/models/project.rb')
| -rw-r--r-- | app/models/project.rb | 134 | 
1 files changed, 84 insertions, 50 deletions
| diff --git a/app/models/project.rb b/app/models/project.rb index 53df29dab02..6ae15a0a50f 100644 --- a/app/models/project.rb +++ b/app/models/project.rb @@ -18,6 +18,7 @@ class Project < ActiveRecord::Base    include SelectForProjectAuthorization    include Routable    include GroupDescendant +  include Gitlab::SQL::Pattern    extend Gitlab::ConfigHelper    extend Gitlab::CurrentSettings @@ -188,7 +189,6 @@ class Project < ActiveRecord::Base    has_one :statistics, class_name: 'ProjectStatistics'    has_one :cluster_project, class_name: 'Clusters::Project' -  has_one :cluster, through: :cluster_project, class_name: 'Clusters::Cluster'    has_many :clusters, through: :cluster_project, class_name: 'Clusters::Cluster'    # Container repositories need to remove data from the container registry, @@ -227,14 +227,13 @@ class Project < ActiveRecord::Base    delegate :members, to: :team, prefix: true    delegate :add_user, :add_users, to: :team    delegate :add_guest, :add_reporter, :add_developer, :add_master, to: :team -  delegate :empty_repo?, to: :repository    # Validations    validates :creator, presence: true, on: :create    validates :description, length: { maximum: 2000 }, allow_blank: true    validates :ci_config_path, -    format: { without: /\.{2}/, -              message: 'cannot include directory traversal.' }, +    format: { without: /(\.{2}|\A\/)/, +              message: 'cannot include leading slash or directory traversal.' },      length: { maximum: 255 },      allow_blank: true    validates :name, @@ -272,8 +271,9 @@ class Project < ActiveRecord::Base    scope :pending_delete, -> { where(pending_delete: true) }    scope :without_deleted, -> { where(pending_delete: false) } -  scope :with_hashed_storage, -> { where('storage_version >= 1') } -  scope :with_legacy_storage, -> { where(storage_version: [nil, 0]) } +  scope :with_storage_feature, ->(feature) { where('storage_version >= :version', version: HASHED_STORAGE_FEATURES[feature]) } +  scope :without_storage_feature, ->(feature) { where('storage_version < :version OR storage_version IS NULL', version: HASHED_STORAGE_FEATURES[feature]) } +  scope :with_unmigrated_storage, -> { where('storage_version < :version OR storage_version IS NULL', version: LATEST_STORAGE_VERSION) }    scope :sorted_by_activity, -> { reorder(last_activity_at: :desc) }    scope :sorted_by_stars, -> { reorder('projects.star_count DESC') } @@ -365,6 +365,7 @@ class Project < ActiveRecord::Base    scope :abandoned, -> { where('projects.last_activity_at < ?', 6.months.ago) }    scope :excluding_project, ->(project) { where.not(id: project) } +  scope :import_started, -> { where(import_status: 'started') }    state_machine :import_status, initial: :none do      event :import_schedule do @@ -423,32 +424,11 @@ class Project < ActiveRecord::Base      #      # query - The search query as a String.      def search(query) -      ptable  = arel_table -      ntable  = Namespace.arel_table -      pattern = "%#{query}%" - -      # unscoping unnecessary conditions that'll be applied -      # when executing `where("projects.id IN (#{union.to_sql})")` -      projects = unscoped.select(:id).where( -        ptable[:path].matches(pattern) -          .or(ptable[:name].matches(pattern)) -          .or(ptable[:description].matches(pattern)) -      ) - -      namespaces = unscoped.select(:id) -        .joins(:namespace) -        .where(ntable[:name].matches(pattern)) - -      union = Gitlab::SQL::Union.new([projects, namespaces]) - -      where("projects.id IN (#{union.to_sql})") # rubocop:disable GitlabSecurity/SqlInjection +      fuzzy_search(query, [:path, :name, :description])      end      def search_by_title(query) -      pattern = "%#{query}%" -      table   = Project.arel_table - -      non_archived.where(table[:name].matches(pattern)) +      non_archived.fuzzy_search(query, [:name])      end      def visibility_levels @@ -518,6 +498,10 @@ class Project < ActiveRecord::Base      auto_devops&.enabled.nil? && !current_application_settings.auto_devops_enabled?    end +  def empty_repo? +    repository.empty? +  end +    def repository_storage_path      Gitlab.config.repositories.storages[repository_storage].try(:[], 'path')    end @@ -580,8 +564,7 @@ class Project < ActiveRecord::Base        if forked?          RepositoryForkWorker.perform_async(id,                                             forked_from_project.repository_storage_path, -                                           forked_from_project.full_path, -                                           self.namespace.full_path) +                                           forked_from_project.disk_path)        else          RepositoryImportWorker.perform_async(self.id)        end @@ -617,7 +600,7 @@ class Project < ActiveRecord::Base    def ci_config_path=(value)      # Strip all leading slashes so that //foo -> foo -    super(value&.sub(%r{\A/+}, '')&.delete("\0")) +    super(value&.delete("\0"))    end    def import_url=(value) @@ -703,10 +686,6 @@ class Project < ActiveRecord::Base      import_type == 'gitea'    end -  def github_import? -    import_type == 'github' -  end -    def check_limit      unless creator.can_create_project? || namespace.kind == 'group'        projects_limit = creator.projects_limit @@ -763,10 +742,10 @@ class Project < ActiveRecord::Base      end    end -  def to_human_reference(from_project = nil) -    if cross_namespace_reference?(from_project) +  def to_human_reference(from = nil) +    if cross_namespace_reference?(from)        name_with_namespace -    elsif cross_project_reference?(from_project) +    elsif cross_project_reference?(from)        name      end    end @@ -775,13 +754,14 @@ class Project < ActiveRecord::Base      Gitlab::Routing.url_helpers.project_url(self)    end -  def new_issue_address(author) +  def new_issuable_address(author, address_type)      return unless Gitlab::IncomingEmail.supports_issue_creation? && author      author.ensure_incoming_email_token! +    suffix = address_type == 'merge_request' ? '+merge-request' : ''      Gitlab::IncomingEmail.reply_address( -      "#{full_path}+#{author.incoming_email_token}") +      "#{full_path}#{suffix}+#{author.incoming_email_token}")    end    def build_commit_note(commit) @@ -919,12 +899,10 @@ class Project < ActiveRecord::Base      @ci_service ||= ci_services.reorder(nil).find_by(active: true)    end -  def deployment_services -    services.where(category: :deployment) -  end - -  def deployment_service -    @deployment_service ||= deployment_services.reorder(nil).find_by(active: true) +  # TODO: This will be extended for multiple enviroment clusters +  def deployment_platform +    @deployment_platform ||= clusters.find_by(enabled: true)&.platform_kubernetes +    @deployment_platform ||= services.where(category: :deployment).reorder(nil).find_by(active: true)    end    def monitoring_services @@ -1046,6 +1024,18 @@ class Project < ActiveRecord::Base      forked_from_project || fork_network&.root_project    end +  def lfs_storage_project +    @lfs_storage_project ||= begin +      result = self + +      # TODO: Make this go to the fork_network root immeadiatly +      # dependant on the discussion in: https://gitlab.com/gitlab-org/gitlab-ce/issues/39769 +      result = result.fork_source while result&.forked? + +      result || self +    end +  end +    def personal?      !group    end @@ -1126,7 +1116,11 @@ class Project < ActiveRecord::Base    end    def project_member(user) -    project_members.find_by(user_id: user) +    if project_members.loaded? +      project_members.find { |member| member.user_id == user.id } +    else +      project_members.find_by(user_id: user) +    end    end    def default_branch @@ -1190,6 +1184,10 @@ class Project < ActiveRecord::Base      !!repository.exists?    end +  def wiki_repository_exists? +    wiki.repository_exists? +  end +    # update visibility_level of forks    def update_forks_visibility_level      return unless visibility_level < visibility_level_was @@ -1433,6 +1431,31 @@ class Project < ActiveRecord::Base      reload_repository!    end +  def after_import +    repository.after_import +    import_finish +    remove_import_jid +    update_project_counter_caches +  end + +  def update_project_counter_caches +    classes = [ +      Projects::OpenIssuesCountService, +      Projects::OpenMergeRequestsCountService +    ] + +    classes.each do |klass| +      klass.new(self).refresh_cache +    end +  end + +  def remove_import_jid +    return unless import_jid + +    Gitlab::SidekiqStatus.unset(import_jid) +    update_column(:import_jid, nil) +  end +    def running_or_pending_build_count(force: false)      Rails.cache.fetch(['projects', id, 'running_or_pending_build_count'], force: force) do        builds.running_or_pending.count(:all) @@ -1528,9 +1551,9 @@ class Project < ActiveRecord::Base    end    def deployment_variables -    return [] unless deployment_service +    return [] unless deployment_platform -    deployment_service.predefined_variables +    deployment_platform.predefined_variables    end    def auto_devops_variables @@ -1690,6 +1713,17 @@ class Project < ActiveRecord::Base      Gitlab::ReferenceCounter.new(gl_repository(is_wiki: wiki))    end +  # Refreshes the expiration time of the associated import job ID. +  # +  # This method can be used by asynchronous importers to refresh the status, +  # preventing the StuckImportJobsWorker from marking the import as failed. +  def refresh_import_jid_expiration +    return unless import_jid + +    Gitlab::SidekiqStatus +      .set(import_jid, StuckImportJobsWorker::IMPORT_JOBS_EXPIRATION) +  end +    private    def storage | 
