123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module Ticket::Search
- extend ActiveSupport::Concern
- # methods defined here are going to extend the class, not the instance of it
- class_methods do
- =begin
- search tickets preferences
- result = Ticket.search_preferences(user_model)
- returns if user has permissions to search
- result = {
- prio: 3000,
- direct_search_index: false
- }
- returns if user has no permissions to search
- result = false
- =end
- def search_preferences(current_user)
- return false if !current_user.permissions?(['ticket.agent', 'ticket.customer'])
- {
- prio: 3000,
- direct_search_index: false,
- }
- end
- =begin
- search tickets via search index
- result = Ticket.search(
- current_user: User.find(123),
- query: 'search something',
- scope: TicketPolicy::ReadScope, # defaults to ReadScope
- limit: 15,
- offset: 100,
- )
- returns
- result = [ticket_model1, ticket_model2]
- search tickets via search index
- result = Ticket.search(
- current_user: User.find(123),
- query: 'search something',
- limit: 15,
- offset: 100,
- full: false,
- )
- returns
- result = [1,3,5,6,7]
- search tickets via database
- result = Ticket.search(
- current_user: User.find(123),
- query: 'some query', # query or condition is required
- scope: TicketPolicy::ReadScope, # defaults to ReadScope
- condition: {
- 'tickets.owner_id' => {
- operator: 'is',
- value: user.id,
- },
- 'tickets.state_id' => {
- operator: 'is',
- value: Ticket::State.where(
- state_type_id: Ticket::StateType.where(
- name: [
- 'pending reminder',
- 'pending action',
- ],
- ).map(&:id),
- ),
- },
- },
- limit: 15,
- offset: 100,
- # sort single column
- sort_by: 'created_at',
- order_by: 'asc',
- # sort multiple columns
- sort_by: [ 'created_at', 'updated_at' ],
- order_by: [ 'asc', 'desc' ],
- full: false,
- )
- returns
- result = [1,3,5,6,7]
- =end
- def search(params)
- # get params
- query = params[:query]
- condition = params[:condition]
- scope = params[:scope] || TicketPolicy::ReadScope
- limit = params[:limit] || 12
- offset = params[:offset] || 0
- current_user = params[:current_user]
- full = false
- if params[:full] == true || params[:full] == 'true' || !params.key?(:full)
- full = true
- end
- sql_helper = ::SqlHelper.new(object: self)
- # check sort
- sort_by = sql_helper.get_sort_by(params, 'updated_at')
- # check order
- order_by = sql_helper.get_order_by(params, 'desc')
- # try search index backend
- if condition.blank? && SearchIndexBackend.enabled?
- query_or = []
- if current_user.permissions?('ticket.agent')
- group_ids = current_user.group_ids_access(scope.const_get(:ACCESS_TYPE))
- if group_ids.present?
- access_condition = {
- 'query_string' => { 'default_field' => 'group_id', 'query' => "\"#{group_ids.join('" OR "')}\"" }
- }
- query_or.push(access_condition)
- end
- end
- if current_user.permissions?('ticket.customer')
- organizations_query = current_user.all_organizations.where(shared: true).map { |o| "organization_id:#{o.id}" }.join(' OR ')
- access_condition = if organizations_query.present?
- {
- 'query_string' => { 'query' => "customer_id:#{current_user.id} OR #{organizations_query}" }
- }
- else
- {
- 'query_string' => { 'default_field' => 'customer_id', 'query' => current_user.id }
- }
- end
- query_or.push(access_condition)
- end
- return [] if query_or.blank?
- query_extension = {
- bool: {
- must: [
- {
- bool: {
- should: query_or,
- },
- },
- ],
- }
- }
- items = SearchIndexBackend.search(query, 'Ticket', limit: limit,
- query_extension: query_extension,
- from: offset,
- sort_by: sort_by,
- order_by: order_by)
- if !full
- return items.pluck(:id)
- end
- tickets = []
- items.each do |item|
- ticket = Ticket.lookup(id: item[:id])
- next if !ticket
- tickets.push ticket
- end
- return tickets
- end
- order_sql = sql_helper.get_order(sort_by, order_by, 'tickets.updated_at DESC')
- tickets_all = scope.new(current_user).resolve
- .reorder(Arel.sql(order_sql))
- .offset(offset)
- .limit(limit)
- ticket_ids = if query
- tickets_all.joins(:articles)
- .where(<<~SQL.squish, query: "%#{SqlHelper.quote_like(query.delete('*'))}%")
- tickets.title LIKE :query
- OR tickets.number LIKE :query
- OR ticket_articles.body LIKE :query
- OR ticket_articles.from LIKE :query
- OR ticket_articles.to LIKE :query
- OR ticket_articles.subject LIKE :query
- SQL
- else
- query_condition, bind_condition, tables = selector2sql(condition)
- tickets_all.joins(tables)
- .where(query_condition, *bind_condition)
- end.group(:id).pluck(:id)
- if full
- ticket_ids.map { |id| Ticket.lookup(id: id) }
- else
- ticket_ids
- end
- end
- end
- end
|