<?php /********************************************************************* ajax.reports.php AJAX interface for reports -- both plot and tabular data are retrievable in JSON format from this utility. Please put plumbing in /scp/ajax.php pattern rules. Jared Hancock <jared@osticket.com> Copyright (c) 2006-2012 osTicket http://www.osticket.com Released under the GNU General Public License WITHOUT ANY WARRANTY. See LICENSE.TXT for details. vim: expandtab sw=4 ts=4 sts=4: **********************************************************************/ if(!defined('INCLUDE_DIR')) die('403'); include_once(INCLUDE_DIR.'class.ticket.php'); /** * Overview Report * * The overview report allows for the display of basic ticket statistics in * both graphical and tabular formats. */ class OverviewReportAjaxAPI extends AjaxController { function enumTabularGroups() { return $this->encode(array("dept"=>"Department", "topic"=>"Topics", # XXX: This will be relative to permissions based on the # logged-in-staff. For basic staff, this will be 'My Stats' "staff"=>"Staff")); } function getData() { $start = $this->get('start', strtotime('last month')); $stop = $this->get('stop', time()); $groups = array( "dept" => array( "table" => DEPT_TABLE, "pk" => "dept_id", "sort" => 'ORDER BY dept_name', "fields" => 'T1.dept_name', "headers" => array('Department') ), "topic" => array( "table" => TOPIC_TABLE, "pk" => "topic_id", "sort" => 'ORDER BY topic', "fields" => "T1.topic", "headers" => array('Help Topic') ), # XXX: This will be relative to permissions based on the # logged-in-staff "staff" => array( "table" => STAFF_TABLE, "pk" => 'staff_id', "sort" => 'ORDER BY T1.lastname, T1.firstname', "fields" => "CONCAT_WS(' ', T1.firstname, T1.lastname)", "headers" => array('Staff Member') ) ); $group = $this->get('group', 'dept'); $info = $groups[$group]; # XXX: Die if $group not in $groups $res = db_query( 'SELECT ' . $info['fields'] . ',' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.status=\'open\') AS Open,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND (A1.staff_id > 0 OR A1.team_id > 0)' .' AND A1.status=\'open\') AS Assigned,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND (A1.staff_id = 0 AND A1.team_id = 0)' .' AND A1.status=\'open\') AS Unassigned,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.isanswered = 0' .' AND A1.status=\'open\') AS Unanswered,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.isoverdue = 1' .' AND A1.status=\'open\') AS Overdue,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.status=\'closed\') AS Closed,' .'(SELECT COUNT(A1.ticket_id) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.reopened is not null) AS Reopened,' .'(SELECT FORMAT(AVG(DATEDIFF(A1.closed, A1.created)),1) FROM '.TICKET_TABLE .' A1 WHERE A1.'.$info['pk'].' = T1.'.$info['pk'] .' AND A1.status=\'closed\') AS ServiceTime' .' FROM ' . $info['table'] . ' T1' ); $rows = array(); while ($row = db_fetch_row($res)) { $rows[] = $row; } return array("columns" => array_merge($info['headers'], array('Open','Assigned','Unassigned','Unanswered', 'Overdue','Closed','Reopened','Service Time')), "data" => $rows); } function getTabularData() { return $this->encode($this->getData()); } function downloadTabularData() { $data = $this->getData(); $csv = '"' . implode('","',$data['columns']) . '"'; foreach ($data['data'] as $row) $csv .= "\n" . '"' . implode('","', $row) . '"'; Http::download( sprintf('%s-report.csv', $this->get('group', 'Department')), 'text/csv', $csv); } function getPlotData() { $start = $this->get('start', 'last month'); $stop = $this->get('stop', 'now'); if (substr($stop, 0, 1) == '+') $stop = $start . $stop; $start = strtotime($start); $stop = strtotime($stop); # Fetch all types of events over the timeframe $res = db_query('SELECT DISTINCT(state) FROM '.TICKET_EVENT_TABLE .' WHERE timestamp BETWEEN FROM_UNIXTIME('.db_input($start) .') AND FROM_UNIXTIME('.db_input($stop) .') ORDER BY 1'); $events = array(); while ($row = db_fetch_row($res)) $events[] = $row[0]; # TODO: Handle user => db timezone offset # XXX: Implement annulled column from the %ticket_event table $res = db_query('SELECT state, DATE_FORMAT(timestamp, \'%Y-%m-%d\'), ' .'COUNT(ticket_id)' .' FROM '.TICKET_EVENT_TABLE .' WHERE timestamp BETWEEN FROM_UNIXTIME('.db_input($start) .') AND FROM_UNIXTIME('.db_input($stop) .') AND NOT annulled' .' GROUP BY state, DATE_FORMAT(timestamp, \'%Y-%m-%d\')' .' ORDER BY 2, 1'); # Initialize array of plot values $plots = array(); foreach ($events as $e) { $plots[$e] = array(); } $time = null; $times = array(); # Iterate over result set, adding zeros for missing ticket events while ($row = db_fetch_row($res)) { $row_time = strtotime($row[1]); if ($time != $row_time) { # New time (and not the first), figure out which events did # not have any tickets associated for this time slot if ($time !== null) { # Not the first record -- add zeros all the arrays that # did not have at least one entry for the timeframe foreach (array_diff($events, $slots) as $slot) $plots[$slot][] = 0; } $slots = array(); $times[] = $time = $row_time; } # Keep track of states for this timeframe $slots[] = $row[0]; $plots[$row[0]][] = (int)$row[2]; } foreach (array_diff($events, $slots) as $slot) $plots[$slot][] = 0; return $this->encode(array("times" => $times, "plots" => $plots, "events"=>$events)); } }