123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class Report::TicketGenericTime
- =begin
- result = Report::TicketGenericTime.aggs(
- range_start: Time.zone.parse('2015-01-01T00:00:00Z'),
- range_end: Time.zone.parse('2015-12-31T23:59:59Z'),
- interval: 'month', # year, quarter, month, week, day, hour, minute, second
- selector: selector, # ticket selector to get only a collection of tickets
- params: { field: 'created_at', selector: selector_sub },
- timezone: 'Europe/Berlin',
- )
- returns
- [4,5,1,5,0,51,5,56,7,4]
- =end
- def self.aggs(params_origin)
- params = params_origin.deep_dup
- interval_es = params[:interval]
- if params[:interval] == 'week'
- interval_es = 'day'
- end
- aggs_interval = {
- from: params[:range_start].iso8601,
- to: params[:range_end].iso8601,
- interval: interval_es, # year, quarter, month, week, day, hour, minute, second
- field: params[:params][:field],
- timezone: params[:timezone],
- }
- without_merged_tickets = {
- 'state.name' => {
- 'operator' => 'is not',
- 'value' => 'merged'
- }
- }
- selector = params[:selector].clone
- if params[:params].present? && params[:params][:selector].present?
- selector = selector.merge(params[:params][:selector])
- end
- selector.merge!(without_merged_tickets) # do not show merged tickets in reports
- result_es = SearchIndexBackend.selectors('Ticket', selector, { current_user: params[:current_user] }, aggs_interval)
- case params[:interval]
- when 'month'
- stop_interval = 12
- when 'week'
- stop_interval = 7
- when 'day'
- stop_interval = ((params[:range_end] - params[:range_start]) / 86_400).to_i + 1
- when 'hour'
- stop_interval = 24
- when 'minute'
- stop_interval = 60
- end
- result = []
- (1..stop_interval).each do |_counter|
- match = false
- if !result_es
- raise "Invalid es result #{result_es.inspect}"
- end
- if !result_es['aggregations']
- raise "Invalid es result, no aggregations #{result_es.inspect}"
- end
- if !result_es['aggregations']['time_buckets']
- raise "Invalid es result, no time_buckets #{result_es.inspect}"
- end
- if !result_es['aggregations']['time_buckets']['buckets']
- raise "Invalid es result, no buckets #{result_es.inspect}"
- end
- result_es['aggregations']['time_buckets']['buckets'].each do |item|
- key_as_string = Time.zone.parse(item['key_as_string'])
- next if !item['doc_count']
- next if item['doc_count'].zero?
- # only compare date - in certain cases elasticsearch timezone offset will not match
- replace = ':\d\dZ$'
- case params[:interval]
- when 'month'
- replace = '\d\dT\d\d:\d\d:\d\dZ$'
- when 'day', 'week'
- replace = '\d\d:\d\d:\d\dZ$'
- end
- next if key_as_string.iso8601.sub(%r{#{replace}}, '') != params[:range_start].iso8601.sub(%r{#{replace}}, '')
- next if match
- match = true
- result.push item['doc_count']
- case params[:interval]
- when 'month'
- params[:range_start] = params[:range_start].next_month
- when 'week', 'day'
- params[:range_start] = params[:range_start].next_day
- when 'hour'
- params[:range_start] = params[:range_start] + 1.hour
- when 'minute'
- params[:range_start] = params[:range_start] + 1.minute
- end
- end
- next if match
- result.push 0
- case params[:interval]
- when 'month'
- params[:range_start] = params[:range_start].next_month
- when 'week'
- params[:range_start] = params[:range_start].next_day
- when 'day'
- params[:range_start] = params[:range_start] + 1.day
- when 'hour'
- params[:range_start] = params[:range_start] + 1.hour
- when 'minute'
- params[:range_start] = params[:range_start] + 1.minute
- end
- end
- result
- end
- =begin
- result = Report::TicketGenericTime.items(
- range_start: Time.zone.parse('2015-01-01T00:00:00Z'),
- range_end: Time.zone.parse('2015-12-31T23:59:59Z'),
- selector: selector, # ticket selector to get only a collection of tickets
- params: { field: 'created_at' },
- )
- returns
- {
- count: 123,
- ticket_ids: [4,5,1,5,0,51,5,56,7,4],
- assets: assets,
- }
- =end
- def self.items(params)
- aggs_interval = {
- from: params[:range_start].iso8601,
- to: params[:range_end].iso8601,
- field: params[:params][:field],
- }
- limit = 6000
- if params[:sheet].blank?
- limit = 100
- end
- without_merged_tickets = {
- 'state.name' => {
- 'operator' => 'is not',
- 'value' => 'merged'
- }
- }
- selector = params[:selector].clone
- if params[:params] && params[:params][:selector]
- selector = selector.merge(params[:params][:selector])
- end
- selector.merge!(without_merged_tickets) # do not show merged tickets in reports
- result = SearchIndexBackend.selectors('Ticket', selector, { current_user: params[:current_user], limit: limit }, aggs_interval)
- result[:ticket_ids] = result.delete(:object_ids)
- return result if params[:sheet].present?
- assets = {}
- result[:ticket_ids].each do |ticket_id|
- suppress(ActiveRecord::RecordNotFound) do
- ticket_full = Ticket.find(ticket_id)
- assets = ticket_full.assets(assets)
- end
- end
- result[:assets] = assets
- result
- end
- end
|