1
0
mirror of https://github.com/meineerde/redmine.git synced 2025-12-19 15:01:14 +00:00

Moved search logic to Redmine::Search (#18631).

git-svn-id: http://svn.redmine.org/redmine/trunk@13769 e93f8b46-1217-0410-a6f0-8f06a7374b81
This commit is contained in:
Jean-Philippe Lang 2014-12-20 08:10:05 +00:00
parent c53eb532c2
commit 71172e2411
2 changed files with 85 additions and 48 deletions

View File

@ -18,12 +18,20 @@
class SearchController < ApplicationController class SearchController < ApplicationController
before_filter :find_optional_project before_filter :find_optional_project
@@search_cache_store ||= ActiveSupport::Cache.lookup_store :memory_store
def index def index
@question = params[:q] || "" @question = params[:q] || ""
@question.strip! @question.strip!
@all_words = params[:all_words] ? params[:all_words].present? : true @all_words = params[:all_words] ? params[:all_words].present? : true
@titles_only = params[:titles_only] ? params[:titles_only].present? : false @titles_only = params[:titles_only] ? params[:titles_only].present? : false
# quick jump to an issue
if (m = @question.match(/^#?(\d+)$/)) && (issue = Issue.visible.find_by_id(m[1].to_i))
redirect_to issue_path(issue)
return
end
projects_to_search = projects_to_search =
case params[:scope] case params[:scope]
when 'all' when 'all'
@ -36,12 +44,6 @@ class SearchController < ApplicationController
@project @project
end end
# quick jump to an issue
if (m = @question.match(/^#?(\d+)$/)) && (issue = Issue.visible.find_by_id(m[1].to_i))
redirect_to issue_path(issue)
return
end
@object_types = Redmine::Search.available_search_types.dup @object_types = Redmine::Search.available_search_types.dup
if projects_to_search.is_a? Project if projects_to_search.is_a? Project
# don't search projects # don't search projects
@ -53,49 +55,18 @@ class SearchController < ApplicationController
@scope = @object_types.select {|t| params[t]} @scope = @object_types.select {|t| params[t]}
@scope = @object_types if @scope.empty? @scope = @object_types if @scope.empty?
# extract tokens from the question fetcher = Redmine::Search::Fetcher.new(
# eg. hello "bye bye" => ["hello", "bye bye"] @question, User.current, @scope, projects_to_search,
@tokens = @question.scan(%r{((\s|^)"[\s\w]+"(\s|$)|\S+)}).collect {|m| m.first.gsub(%r{(^\s*"\s*|\s*"\s*$)}, '')} :all_words => @all_words, :titles_only => @titles_only
# tokens must be at least 2 characters long
@tokens = @tokens.uniq.select {|w| w.length > 1 }
if !@tokens.empty?
# no more than 5 tokens to search for
@tokens.slice! 5..-1 if @tokens.size > 5
limit = 10
@result_count = 0
@result_count_by_type = Hash.new {|h,k| h[k] = 0}
ranks_and_ids = []
# get all the results ranks and ids
@scope.each do |scope|
klass = scope.singularize.camelcase.constantize
ranks_and_ids_in_scope = klass.search_result_ranks_and_ids(@tokens, User.current, projects_to_search,
:all_words => @all_words,
:titles_only => @titles_only
) )
@result_count_by_type[scope] += ranks_and_ids_in_scope.size
@result_count += ranks_and_ids_in_scope.size
ranks_and_ids += ranks_and_ids_in_scope.map {|r| [scope, r]}
end
@result_pages = Paginator.new @result_count, limit, params['page']
# sort results, higher rank and id first if fetcher.tokens.present?
ranks_and_ids.sort! {|a,b| b.last <=> a.last } @result_count = fetcher.result_count
ranks_and_ids = ranks_and_ids[@result_pages.offset, limit] || [] @result_count_by_type = fetcher.result_count_by_type
@tokens = fetcher.tokens
# load the results to display @result_pages = Paginator.new @result_count, 10, params['page']
results_by_scope = Hash.new {|h,k| h[k] = []} @results = fetcher.results(@result_pages.offset, @result_pages.per_page)
ranks_and_ids.group_by(&:first).each do |scope, rs|
klass = scope.singularize.camelcase.constantize
results_by_scope[scope] += klass.search_results_from_ids(rs.map(&:last).map(&:last))
end
@results = ranks_and_ids.map do |scope, r|
results_by_scope[scope].detect {|record| record.id == r.last}
end.compact
else else
@question = "" @question = ""
end end

View File

@ -19,7 +19,6 @@ module Redmine
module Search module Search
mattr_accessor :available_search_types mattr_accessor :available_search_types
@@available_search_types = [] @@available_search_types = []
class << self class << self
@ -34,6 +33,73 @@ module Redmine
end end
end end
class Fetcher
attr_reader :tokens
def initialize(question, user, scope, projects, options={})
@user = user
@question = question.strip
@scope = scope
@projects = projects
@options = options
# extract tokens from the question
# eg. hello "bye bye" => ["hello", "bye bye"]
@tokens = @question.scan(%r{((\s|^)"[\s\w]+"(\s|$)|\S+)}).collect {|m| m.first.gsub(%r{(^\s*"\s*|\s*"\s*$)}, '')}
# tokens must be at least 2 characters long
@tokens = @tokens.uniq.select {|w| w.length > 1 }
# no more than 5 tokens to search for
@tokens.slice! 5..-1
end
def result_count
result_ids.size
end
def result_count_by_type
ret = Hash.new {|h,k| h[k] = 0}
result_ids.group_by(&:first).each do |scope, ids|
ret[scope] += ids.size
end
ret
end
def results(offset, limit)
result_ids_to_load = result_ids[offset, limit] || []
results_by_scope = Hash.new {|h,k| h[k] = []}
result_ids_to_load.group_by(&:first).each do |scope, scope_and_ids|
klass = scope.singularize.camelcase.constantize
results_by_scope[scope] += klass.search_results_from_ids(scope_and_ids.map(&:last))
end
result_ids_to_load.map do |scope, id|
results_by_scope[scope].detect {|record| record.id == id}
end.compact
end
def result_ids
@ranks_and_ids ||= load_result_ids
end
private
def load_result_ids
ret = []
# get all the results ranks and ids
@scope.each do |scope|
klass = scope.singularize.camelcase.constantize
ranks_and_ids_in_scope = klass.search_result_ranks_and_ids(@tokens, User.current, @projects, @options)
# converts timestamps to integers for faster sort
ret += ranks_and_ids_in_scope.map {|rank, id| [scope, [rank.to_i, id]]}
end
# sort results, higher rank and id first
ret.sort! {|a,b| b.last <=> a.last}
ret.map! {|scope, r| [scope, r.last]}
ret
end
end
module Controller module Controller
def self.included(base) def self.included(base)
base.extend(ClassMethods) base.extend(ClassMethods)