123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class ReportsController < ApplicationController
- prepend_before_action :authenticate_and_authorize!
- # GET /api/reports/config
- def reporting_config
- if !Report.enabled?
- render json: {
- error: __('Elasticsearch needs to be configured!'),
- }
- return
- end
- render json: {
- config: Report.config,
- profiles: Report::Profile.list,
- }
- end
- # GET /api/reports/generate
- def generate
- get_params = params_all
- return if !get_params
- result = {}
- get_params[:metric][:backend].each do |backend|
- condition = get_params[:profile].condition
- if backend[:condition]
- backend[:condition].merge(condition)
- else
- backend[:condition] = condition
- end
- next if !backend[:adapter]
- next if params['backends'][backend[:name]].blank?
- result[backend[:name]] = backend[:adapter].aggs(
- range_start: get_params[:start],
- range_end: get_params[:stop],
- interval: get_params[:range],
- selector: backend[:condition],
- params: backend[:params],
- timezone: get_params[:timezone],
- timezone_offset: get_params[:timezone_offset],
- current_user: current_user
- )
- end
- render json: {
- data: result
- }
- end
- # GET /api/reports/sets
- def sets
- get_params = params_all
- return if !get_params
- if !params[:downloadBackendSelected]
- render json: {
- error: __("Required parameter 'downloadBackendSelected' is missing."),
- }, status: :unprocessable_entity
- return
- end
- # get data
- result = {}
- excel = nil
- filename = nil
- get_params[:metric][:backend].each do |backend|
- next if params[:downloadBackendSelected] != backend[:name]
- condition = get_params[:profile].condition
- if backend[:condition]
- backend[:condition].merge(condition)
- else
- backend[:condition] = condition
- end
- next if !backend[:adapter]
- result = backend[:adapter].items(
- range_start: get_params[:start],
- range_end: get_params[:stop],
- interval: get_params[:range],
- selector: backend[:condition],
- params: backend[:params],
- sheet: params[:sheet],
- timezone: get_params[:timezone],
- timezone_offset: get_params[:timezone_offset],
- current_user: current_user
- )
- result = { count: 0, ticket_ids: [] } if result.nil?
- # generate sheet
- if params[:sheet]
- excel = ExcelSheet::Ticket.new(
- title: "#{get_params[:profile].name} (#{backend[:display]})",
- ticket_ids: result[:ticket_ids],
- timezone: params[:timezone],
- locale: current_user.locale,
- )
- filename = "tickets-#{get_params[:profile].name}-#{backend[:display]}.xlsx"
- end
- break
- end
- if excel
- send_data(
- excel.content,
- filename: filename,
- type: ExcelSheet::CONTENT_TYPE,
- disposition: 'attachment'
- )
- return
- end
- render json: result
- end
- def params_all
- if !params[:profiles] && !params[:profile_id]
- raise Exceptions::UnprocessableEntity, __("Required parameter 'profile' is missing.")
- end
- if params[:profile_id]
- profile = Report::Profile.find(params[:profile_id])
- else
- params[:profiles].each do |profile_id, active|
- next if !active
- profile = Report::Profile.find(profile_id)
- end
- end
- if !profile
- raise Exceptions::UnprocessableEntity, __('The active reporting profile could not be found.')
- end
- local_config = Report.config
- if !local_config || !local_config[:metric] || !local_config[:metric][params[:metric].to_sym]
- raise Exceptions::UnprocessableEntity, "Could not find metric #{params[:metric]}"
- end
- metric = local_config[:metric][params[:metric].to_sym]
- case params[:timeRange]
- when 'realtime'
- start_at = 60.minutes.ago
- stop_at = Time.zone.now
- range = 'minute'
- when 'day'
- date = Date.parse("#{params[:year]}-#{params[:month]}-#{params[:day]}").to_s
- start_at = Time.zone.parse("#{date}T00:00:00Z")
- stop_at = Time.zone.parse("#{date}T23:59:59Z")
- range = 'hour'
- when 'week'
- start_week_at = Date.commercial(params[:year].to_i, params[:week].to_i)
- stop_week_at = start_week_at.end_of_week
- start_at = Time.zone.parse("#{start_week_at.year}-#{start_week_at.month}-#{start_week_at.day}T00:00:00Z")
- stop_at = Time.zone.parse("#{stop_week_at.year}-#{stop_week_at.month}-#{stop_week_at.day}T23:59:59Z")
- range = 'week'
- when 'month'
- start_at = Time.zone.parse("#{params[:year]}-#{params[:month]}-01T00:00:00Z")
- stop_at = Time.zone.parse("#{params[:year]}-#{params[:month]}-#{start_at.end_of_month.day}T23:59:59Z")
- range = 'day'
- else
- start_at = Time.zone.parse("#{params[:year]}-01-01T00:00:00Z")
- stop_at = Time.zone.parse("#{params[:year]}-12-31T23:59:59Z")
- range = 'month'
- end
- params[:timezone] ||= Setting.get('timezone_default')
- if params[:timeRange] != 'realtime'
- offset = stop_at.in_time_zone(params[:timezone]).utc_offset
- start_at -= offset
- stop_at -= offset
- end
- {
- profile: profile,
- metric: metric,
- config: local_config,
- start: start_at,
- stop: stop_at,
- range: range,
- timezone: params[:timezone],
- timezone_offset: offset,
- }
- end
- end
|