search_controller.rb 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. # Copyright (C) 2012-2016 Zammad Foundation, http://zammad-foundation.org/
  2. class SearchController < ApplicationController
  3. prepend_before_action :authentication_check
  4. # GET|POST /api/v1/search
  5. # GET|POST /api/v1/search/:objects
  6. def search_generic
  7. # enable search only for users with valid session
  8. if !current_user
  9. response_access_deny
  10. return true
  11. end
  12. # get params
  13. query = params[:query]
  14. if query.respond_to?(:permit!)
  15. query = query.permit!.to_h
  16. end
  17. limit = params[:limit] || 10
  18. # convert objects string into array of class names
  19. # e.g. user-ticket-another_object = %w( User Ticket AnotherObject )
  20. objects = if !params[:objects]
  21. Setting.get('models_searchable')
  22. else
  23. params[:objects].split('-').map(&:camelize)
  24. end
  25. # get priorities of result
  26. objects_in_order = []
  27. objects_in_order_hash = {}
  28. objects.each do |object|
  29. preferences = object.constantize.search_preferences(current_user)
  30. next if !preferences
  31. objects_in_order_hash[preferences[:prio]] = object
  32. end
  33. objects_in_order_hash.keys.sort.reverse_each do |prio|
  34. objects_in_order.push objects_in_order_hash[prio]
  35. end
  36. # try search index backend
  37. assets = {}
  38. result = []
  39. if SearchIndexBackend.enabled?
  40. # get direct search index based objects
  41. objects_with_direct_search_index = []
  42. objects_without_direct_search_index = []
  43. objects.each do |object|
  44. preferences = object.constantize.search_preferences(current_user)
  45. next if !preferences
  46. if preferences[:direct_search_index]
  47. objects_with_direct_search_index.push object
  48. else
  49. objects_without_direct_search_index.push object
  50. end
  51. end
  52. # do only one query to index search backend
  53. if objects_with_direct_search_index.present?
  54. items = SearchIndexBackend.search(query, limit, objects_with_direct_search_index)
  55. items.each do |item|
  56. require item[:type].to_filename
  57. record = Kernel.const_get(item[:type]).lookup(id: item[:id])
  58. next if !record
  59. assets = record.assets(assets)
  60. result.push item
  61. end
  62. end
  63. # e. g. do ticket query by Ticket class to handle ticket permissions
  64. objects_without_direct_search_index.each do |object|
  65. object_result = search_generic_backend(object, query, limit, current_user, assets)
  66. if object_result.present?
  67. result = result.concat(object_result)
  68. end
  69. end
  70. # sort order by object priority
  71. result_in_order = []
  72. objects_in_order.each do |object|
  73. result.each do |item|
  74. next if item[:type] != object
  75. item[:id] = item[:id].to_i
  76. result_in_order.push item
  77. end
  78. end
  79. result = result_in_order
  80. else
  81. # do query
  82. objects_in_order.each do |object|
  83. object_result = search_generic_backend(object, query, limit, current_user, assets)
  84. if object_result.present?
  85. result = result.concat(object_result)
  86. end
  87. end
  88. end
  89. render json: {
  90. assets: assets,
  91. result: result,
  92. }
  93. end
  94. private
  95. def search_generic_backend(object, query, limit, current_user, assets)
  96. found_objects = object.constantize.search(
  97. query: query,
  98. limit: limit,
  99. current_user: current_user,
  100. )
  101. result = []
  102. found_objects.each do |found_object|
  103. item = {
  104. id: found_object.id,
  105. type: found_object.class.to_s
  106. }
  107. result.push item
  108. assets = found_object.assets(assets)
  109. end
  110. result
  111. end
  112. end