2018-12-13 13:39:08 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
module Gitlab
|
|
|
|
class SearchResults
|
2017-08-17 22:00:37 +05:30
|
|
|
class FoundBlob
|
2018-03-17 18:26:18 +05:30
|
|
|
include EncodingHelper
|
2018-12-13 13:39:08 +05:30
|
|
|
include Presentable
|
|
|
|
include BlobLanguageFromGitAttributes
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
attr_reader :id, :filename, :basename, :ref, :startline, :data, :project
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
def initialize(opts = {})
|
|
|
|
@id = opts.fetch(:id, nil)
|
2018-03-17 18:26:18 +05:30
|
|
|
@filename = encode_utf8(opts.fetch(:filename, nil))
|
|
|
|
@basename = encode_utf8(opts.fetch(:basename, nil))
|
2017-08-17 22:00:37 +05:30
|
|
|
@ref = opts.fetch(:ref, nil)
|
|
|
|
@startline = opts.fetch(:startline, nil)
|
2018-03-17 18:26:18 +05:30
|
|
|
@data = encode_utf8(opts.fetch(:data, nil))
|
|
|
|
@per_page = opts.fetch(:per_page, 20)
|
2018-12-13 13:39:08 +05:30
|
|
|
@project = opts.fetch(:project, nil)
|
|
|
|
# Some caller does not have project object (e.g. elastic search),
|
|
|
|
# yet they can trigger many calls in one go,
|
|
|
|
# causing duplicated queries.
|
|
|
|
# Allow those to just pass project_id instead.
|
2018-03-17 18:26:18 +05:30
|
|
|
@project_id = opts.fetch(:project_id, nil)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def path
|
|
|
|
filename
|
|
|
|
end
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
def project_id
|
|
|
|
@project_id || @project&.id
|
|
|
|
end
|
|
|
|
|
|
|
|
def present
|
|
|
|
super(presenter_class: BlobPresenter)
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
attr_reader :current_user, :query, :per_page
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
# Limit search results by passed projects
|
2015-04-26 12:48:37 +05:30
|
|
|
# It allows us to search only for projects user has access to
|
2016-06-02 11:05:42 +05:30
|
|
|
attr_reader :limit_projects
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
# Whether a custom filter is used to restrict scope of projects.
|
|
|
|
# If the default filter (which lists all projects user has access to)
|
|
|
|
# is used, we can skip it when filtering merge requests and optimize the
|
|
|
|
# query
|
|
|
|
attr_reader :default_project_filter
|
|
|
|
|
|
|
|
def initialize(current_user, limit_projects, query, default_project_filter: false, per_page: 20)
|
2016-06-02 11:05:42 +05:30
|
|
|
@current_user = current_user
|
|
|
|
@limit_projects = limit_projects || Project.all
|
2018-03-17 18:26:18 +05:30
|
|
|
@query = query
|
|
|
|
@default_project_filter = default_project_filter
|
|
|
|
@per_page = per_page
|
|
|
|
end
|
|
|
|
|
|
|
|
def objects(scope, page = nil, without_count = true)
|
|
|
|
collection = case scope
|
|
|
|
when 'projects'
|
|
|
|
projects.page(page).per(per_page)
|
|
|
|
when 'issues'
|
|
|
|
issues.page(page).per(per_page)
|
|
|
|
when 'merge_requests'
|
|
|
|
merge_requests.page(page).per(per_page)
|
|
|
|
when 'milestones'
|
|
|
|
milestones.page(page).per(per_page)
|
|
|
|
else
|
|
|
|
Kaminari.paginate_array([]).page(page).per(per_page)
|
|
|
|
end
|
|
|
|
|
|
|
|
without_count ? collection.without_count : collection
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
def limited_projects_count
|
|
|
|
@limited_projects_count ||= projects.limit(count_limit).count
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
def limited_issues_count
|
|
|
|
return @limited_issues_count if @limited_issues_count
|
|
|
|
|
|
|
|
# By default getting limited count (e.g. 1000+) is fast on issuable
|
|
|
|
# collections except for issues, where filtering both not confidential
|
|
|
|
# and confidential issues user has access to, is too complex.
|
|
|
|
# It's faster to try to fetch all public issues first, then only
|
|
|
|
# if necessary try to fetch all issues.
|
|
|
|
sum = issues(public_only: true).limit(count_limit).count
|
|
|
|
@limited_issues_count = sum < count_limit ? issues.limit(count_limit).count : sum
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
def limited_merge_requests_count
|
|
|
|
@limited_merge_requests_count ||= merge_requests.limit(count_limit).count
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
def limited_milestones_count
|
|
|
|
@limited_milestones_count ||= milestones.limit(count_limit).count
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
def single_commit_result?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
def count_limit
|
|
|
|
1001
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
private
|
|
|
|
|
|
|
|
def projects
|
2016-06-02 11:05:42 +05:30
|
|
|
limit_projects.search(query)
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2018-03-17 18:26:18 +05:30
|
|
|
def issues(finder_params = {})
|
|
|
|
issues = IssuesFinder.new(current_user, finder_params).execute
|
|
|
|
unless default_project_filter
|
|
|
|
issues = issues.where(project_id: project_ids_relation)
|
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
issues =
|
|
|
|
if query =~ /#(\d+)\z/
|
|
|
|
issues.where(iid: $1)
|
|
|
|
else
|
|
|
|
issues.full_search(query)
|
|
|
|
end
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
issues.reorder('updated_at DESC')
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2015-04-26 12:48:37 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2015-09-25 12:07:36 +05:30
|
|
|
def milestones
|
2016-06-02 11:05:42 +05:30
|
|
|
milestones = Milestone.where(project_id: project_ids_relation)
|
2015-09-25 12:07:36 +05:30
|
|
|
milestones = milestones.search(query)
|
2018-03-17 18:26:18 +05:30
|
|
|
milestones.reorder('updated_at DESC')
|
2015-09-25 12:07:36 +05:30
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2015-09-25 12:07:36 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2015-04-26 12:48:37 +05:30
|
|
|
def merge_requests
|
2018-03-17 18:26:18 +05:30
|
|
|
merge_requests = MergeRequestsFinder.new(current_user).execute
|
|
|
|
unless default_project_filter
|
|
|
|
merge_requests = merge_requests.in_projects(project_ids_relation)
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
merge_requests =
|
|
|
|
if query =~ /[#!](\d+)\z/
|
|
|
|
merge_requests.where(iid: $1)
|
|
|
|
else
|
|
|
|
merge_requests.full_search(query)
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
merge_requests.reorder('updated_at DESC')
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2015-04-26 12:48:37 +05:30
|
|
|
|
|
|
|
def default_scope
|
|
|
|
'projects'
|
|
|
|
end
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-06-02 11:05:42 +05:30
|
|
|
def project_ids_relation
|
|
|
|
limit_projects.select(:id).reorder(nil)
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
end
|