The index is already added in production.
Adds a new reporting API that returns conversation counts grouped by
channel type and first response time buckets (0-1h, 1-4h, 4-8h, 8-24h,
24h+).
- GET /api/v2/accounts/:id/reports/first_response_time_distribution
- Uses SQL aggregation to handle large datasets efficiently
- Adds composite index on reporting_events for query performance
Tested on production workload.
Request: GET
`/api/v2/accounts/1/reports/first_response_time_distribution?since=<since>&until=<until>`
Response payload:
```
{
"Channel::WebWidget": {
"0-1h": 120,
"1-4h": 85,
"4-8h": 32,
"8-24h": 12,
"24h+": 3
},
"Channel::Email": {
"0-1h": 12,
"1-4h": 28,
"4-8h": 45,
"8-24h": 35,
"24h+": 10
},
"Channel::FacebookPage": {
"0-1h": 50,
"1-4h": 30,
"4-8h": 15,
"8-24h": 8,
"24h+": 2
}
}
```
---------
Co-authored-by: Muhsin Keloth <muhsinkeramam@gmail.com>
175 lines
4.6 KiB
Ruby
175 lines
4.6 KiB
Ruby
class Api::V2::Accounts::ReportsController < Api::V1::Accounts::BaseController
|
|
include Api::V2::Accounts::ReportsHelper
|
|
include Api::V2::Accounts::HeatmapHelper
|
|
|
|
before_action :check_authorization
|
|
|
|
def index
|
|
builder = V2::Reports::Conversations::ReportBuilder.new(Current.account, report_params)
|
|
data = builder.timeseries
|
|
render json: data
|
|
end
|
|
|
|
def summary
|
|
render json: build_summary(:summary)
|
|
end
|
|
|
|
def bot_summary
|
|
render json: build_summary(:bot_summary)
|
|
end
|
|
|
|
def agents
|
|
@report_data = generate_agents_report
|
|
generate_csv('agents_report', 'api/v2/accounts/reports/agents')
|
|
end
|
|
|
|
def inboxes
|
|
@report_data = generate_inboxes_report
|
|
generate_csv('inboxes_report', 'api/v2/accounts/reports/inboxes')
|
|
end
|
|
|
|
def labels
|
|
@report_data = generate_labels_report
|
|
generate_csv('labels_report', 'api/v2/accounts/reports/labels')
|
|
end
|
|
|
|
def teams
|
|
@report_data = generate_teams_report
|
|
generate_csv('teams_report', 'api/v2/accounts/reports/teams')
|
|
end
|
|
|
|
def conversations_summary
|
|
@report_data = generate_conversations_report
|
|
generate_csv('conversations_summary_report', 'api/v2/accounts/reports/conversations_summary')
|
|
end
|
|
|
|
def conversation_traffic
|
|
@report_data = generate_conversations_heatmap_report
|
|
timezone_offset = (params[:timezone_offset] || 0).to_f
|
|
@timezone = ActiveSupport::TimeZone[timezone_offset]
|
|
|
|
generate_csv('conversation_traffic_reports', 'api/v2/accounts/reports/conversation_traffic')
|
|
end
|
|
|
|
def conversations
|
|
return head :unprocessable_entity if params[:type].blank?
|
|
|
|
render json: conversation_metrics
|
|
end
|
|
|
|
def bot_metrics
|
|
bot_metrics = V2::Reports::BotMetricsBuilder.new(Current.account, params).metrics
|
|
render json: bot_metrics
|
|
end
|
|
|
|
def inbox_label_matrix
|
|
builder = V2::Reports::InboxLabelMatrixBuilder.new(
|
|
account: Current.account,
|
|
params: inbox_label_matrix_params
|
|
)
|
|
render json: builder.build
|
|
end
|
|
|
|
def first_response_time_distribution
|
|
builder = V2::Reports::FirstResponseTimeDistributionBuilder.new(
|
|
account: Current.account,
|
|
params: first_response_time_distribution_params
|
|
)
|
|
render json: builder.build
|
|
end
|
|
|
|
private
|
|
|
|
def generate_csv(filename, template)
|
|
response.headers['Content-Type'] = 'text/csv'
|
|
response.headers['Content-Disposition'] = "attachment; filename=#{filename}.csv"
|
|
render layout: false, template: template, formats: [:csv]
|
|
end
|
|
|
|
def check_authorization
|
|
authorize :report, :view?
|
|
end
|
|
|
|
def common_params
|
|
{
|
|
type: params[:type].to_sym,
|
|
id: params[:id],
|
|
group_by: params[:group_by],
|
|
business_hours: ActiveModel::Type::Boolean.new.cast(params[:business_hours])
|
|
}
|
|
end
|
|
|
|
def current_summary_params
|
|
common_params.merge({
|
|
since: range[:current][:since],
|
|
until: range[:current][:until],
|
|
timezone_offset: params[:timezone_offset]
|
|
})
|
|
end
|
|
|
|
def previous_summary_params
|
|
common_params.merge({
|
|
since: range[:previous][:since],
|
|
until: range[:previous][:until],
|
|
timezone_offset: params[:timezone_offset]
|
|
})
|
|
end
|
|
|
|
def report_params
|
|
common_params.merge({
|
|
metric: params[:metric],
|
|
since: params[:since],
|
|
until: params[:until],
|
|
timezone_offset: params[:timezone_offset]
|
|
})
|
|
end
|
|
|
|
def conversation_params
|
|
{
|
|
type: params[:type].to_sym,
|
|
user_id: params[:user_id],
|
|
page: params[:page].presence || 1
|
|
}
|
|
end
|
|
|
|
def range
|
|
{
|
|
current: {
|
|
since: params[:since],
|
|
until: params[:until]
|
|
},
|
|
previous: {
|
|
since: (params[:since].to_i - (params[:until].to_i - params[:since].to_i)).to_s,
|
|
until: params[:since]
|
|
}
|
|
}
|
|
end
|
|
|
|
def build_summary(method)
|
|
builder = V2::Reports::Conversations::MetricBuilder
|
|
current_summary = builder.new(Current.account, current_summary_params).send(method)
|
|
previous_summary = builder.new(Current.account, previous_summary_params).send(method)
|
|
current_summary.merge(previous: previous_summary)
|
|
end
|
|
|
|
def conversation_metrics
|
|
V2::ReportBuilder.new(Current.account, conversation_params).conversation_metrics
|
|
end
|
|
|
|
def inbox_label_matrix_params
|
|
{
|
|
since: params[:since],
|
|
until: params[:until],
|
|
inbox_ids: params[:inbox_ids],
|
|
label_ids: params[:label_ids]
|
|
}
|
|
end
|
|
|
|
def first_response_time_distribution_params
|
|
{
|
|
since: params[:since],
|
|
until: params[:until]
|
|
}
|
|
end
|
|
end
|