You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
803 lines
18 KiB
803 lines
18 KiB
import moment from 'moment';
|
|
|
|
import { LivechatRooms } from '../../../models';
|
|
|
|
/**
|
|
* return readable time format from seconds
|
|
* @param {Double} sec seconds
|
|
* @return {String} Readable string format
|
|
*/
|
|
const secondsToHHMMSS = (sec) => {
|
|
sec = parseFloat(sec);
|
|
|
|
let hours = Math.floor(sec / 3600);
|
|
let minutes = Math.floor((sec - (hours * 3600)) / 60);
|
|
let seconds = Math.round(sec - (hours * 3600) - (minutes * 60));
|
|
|
|
if (hours < 10) { hours = `0${ hours }`; }
|
|
if (minutes < 10) { minutes = `0${ minutes }`; }
|
|
if (seconds < 10) { seconds = `0${ seconds }`; }
|
|
|
|
if (hours > 0) {
|
|
return `${ hours }:${ minutes }:${ seconds }`;
|
|
}
|
|
if (minutes > 0) {
|
|
return `${ minutes }:${ seconds }`;
|
|
}
|
|
return sec;
|
|
};
|
|
|
|
export const Analytics = {
|
|
getAgentOverviewData(options) {
|
|
const from = moment(options.daterange.from);
|
|
const to = moment(options.daterange.to);
|
|
|
|
if (!(moment(from).isValid() && moment(to).isValid())) {
|
|
console.log('livechat:getAgentOverviewData => Invalid dates');
|
|
return;
|
|
}
|
|
|
|
if (!this.AgentOverviewData[options.chartOptions.name]) {
|
|
console.log(`Method RocketChat.Livechat.Analytics.AgentOverviewData.${ options.chartOptions.name } does NOT exist`);
|
|
return;
|
|
}
|
|
|
|
return this.AgentOverviewData[options.chartOptions.name](from, to);
|
|
},
|
|
|
|
getAnalyticsChartData(options) {
|
|
// Check if function exists, prevent server error in case property altered
|
|
if (!this.ChartData[options.chartOptions.name]) {
|
|
console.log(`Method RocketChat.Livechat.Analytics.ChartData.${ options.chartOptions.name } does NOT exist`);
|
|
return;
|
|
}
|
|
|
|
const from = moment(options.daterange.from);
|
|
const to = moment(options.daterange.to);
|
|
|
|
if (!(moment(from).isValid() && moment(to).isValid())) {
|
|
console.log('livechat:getAnalyticsChartData => Invalid dates');
|
|
return;
|
|
}
|
|
|
|
|
|
const data = {
|
|
chartLabel: options.chartOptions.name,
|
|
dataLabels: [],
|
|
dataPoints: [],
|
|
};
|
|
|
|
if (from.diff(to) === 0) { // data for single day
|
|
for (let m = moment(from); m.diff(to, 'days') <= 0; m.add(1, 'hours')) {
|
|
const hour = m.format('H');
|
|
data.dataLabels.push(`${ moment(hour, ['H']).format('hA') }-${ moment((parseInt(hour) + 1) % 24, ['H']).format('hA') }`);
|
|
|
|
const date = {
|
|
gte: m,
|
|
lt: moment(m).add(1, 'hours'),
|
|
};
|
|
|
|
data.dataPoints.push(this.ChartData[options.chartOptions.name](date));
|
|
}
|
|
} else {
|
|
for (let m = moment(from); m.diff(to, 'days') <= 0; m.add(1, 'days')) {
|
|
data.dataLabels.push(m.format('M/D'));
|
|
|
|
const date = {
|
|
gte: m,
|
|
lt: moment(m).add(1, 'days'),
|
|
};
|
|
|
|
data.dataPoints.push(this.ChartData[options.chartOptions.name](date));
|
|
}
|
|
}
|
|
|
|
return data;
|
|
},
|
|
|
|
getAnalyticsOverviewData(options) {
|
|
const from = moment(options.daterange.from);
|
|
const to = moment(options.daterange.to);
|
|
|
|
if (!(moment(from).isValid() && moment(to).isValid())) {
|
|
console.log('livechat:getAnalyticsOverviewData => Invalid dates');
|
|
return;
|
|
}
|
|
|
|
if (!this.OverviewData[options.analyticsOptions.name]) {
|
|
console.log(`Method RocketChat.Livechat.Analytics.OverviewData.${ options.analyticsOptions.name } does NOT exist`);
|
|
return;
|
|
}
|
|
|
|
return this.OverviewData[options.analyticsOptions.name](from, to);
|
|
},
|
|
|
|
ChartData: {
|
|
/**
|
|
*
|
|
* @param {Object} date {gte: {Date}, lt: {Date}}
|
|
*
|
|
* @returns {Integer}
|
|
*/
|
|
Total_conversations(date) {
|
|
return LivechatRooms.getTotalConversationsBetweenDate('l', date);
|
|
},
|
|
|
|
Avg_chat_duration(date) {
|
|
let total = 0;
|
|
let count = 0;
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ metrics }) => {
|
|
if (metrics && metrics.chatDuration) {
|
|
total += metrics.chatDuration;
|
|
count++;
|
|
}
|
|
});
|
|
|
|
const avgCD = count ? total / count : 0;
|
|
return Math.round(avgCD * 100) / 100;
|
|
},
|
|
|
|
Total_messages(date) {
|
|
let total = 0;
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ msgs }) => {
|
|
if (msgs) {
|
|
total += msgs;
|
|
}
|
|
});
|
|
|
|
return total;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Object} date {gte: {Date}, lt: {Date}}
|
|
*
|
|
* @returns {Double}
|
|
*/
|
|
Avg_first_response_time(date) {
|
|
let frt = 0;
|
|
let count = 0;
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ metrics }) => {
|
|
if (metrics && metrics.response && metrics.response.ft) {
|
|
frt += metrics.response.ft;
|
|
count++;
|
|
}
|
|
});
|
|
|
|
const avgFrt = count ? frt / count : 0;
|
|
return Math.round(avgFrt * 100) / 100;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Object} date {gte: {Date}, lt: {Date}}
|
|
*
|
|
* @returns {Double}
|
|
*/
|
|
Best_first_response_time(date) {
|
|
let maxFrt;
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ metrics }) => {
|
|
if (metrics && metrics.response && metrics.response.ft) {
|
|
maxFrt = maxFrt ? Math.min(maxFrt, metrics.response.ft) : metrics.response.ft;
|
|
}
|
|
});
|
|
|
|
if (!maxFrt) { maxFrt = 0; }
|
|
|
|
return Math.round(maxFrt * 100) / 100;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Object} date {gte: {Date}, lt: {Date}}
|
|
*
|
|
* @returns {Double}
|
|
*/
|
|
Avg_response_time(date) {
|
|
let art = 0;
|
|
let count = 0;
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ metrics }) => {
|
|
if (metrics && metrics.response && metrics.response.avg) {
|
|
art += metrics.response.avg;
|
|
count++;
|
|
}
|
|
});
|
|
|
|
const avgArt = count ? art / count : 0;
|
|
|
|
return Math.round(avgArt * 100) / 100;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Object} date {gte: {Date}, lt: {Date}}
|
|
*
|
|
* @returns {Double}
|
|
*/
|
|
Avg_reaction_time(date) {
|
|
let arnt = 0;
|
|
let count = 0;
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({ metrics }) => {
|
|
if (metrics && metrics.reaction && metrics.reaction.ft) {
|
|
arnt += metrics.reaction.ft;
|
|
count++;
|
|
}
|
|
});
|
|
|
|
const avgArnt = count ? arnt / count : 0;
|
|
|
|
return Math.round(avgArnt * 100) / 100;
|
|
},
|
|
},
|
|
|
|
OverviewData: {
|
|
/**
|
|
*
|
|
* @param {Map} map
|
|
*
|
|
* @return {String}
|
|
*/
|
|
getKeyHavingMaxValue(map, def) {
|
|
let maxValue = 0;
|
|
let maxKey = def; // default
|
|
|
|
map.forEach((value, key) => {
|
|
if (value > maxValue) {
|
|
maxValue = value;
|
|
maxKey = key;
|
|
}
|
|
});
|
|
|
|
return maxKey;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array[Object]}
|
|
*/
|
|
Conversations(from, to) {
|
|
let totalConversations = 0; // Total conversations
|
|
let openConversations = 0; // open conversations
|
|
let totalMessages = 0; // total msgs
|
|
const totalMessagesOnWeekday = new Map(); // total messages on weekdays i.e Monday, Tuesday...
|
|
const totalMessagesInHour = new Map(); // total messages in hour 0, 1, ... 23 of weekday
|
|
const days = to.diff(from, 'days') + 1; // total days
|
|
|
|
const summarize = (m) => ({ metrics, msgs }) => {
|
|
if (metrics && !metrics.chatDuration) {
|
|
openConversations++;
|
|
}
|
|
totalMessages += msgs;
|
|
|
|
const weekday = m.format('dddd'); // @string: Monday, Tuesday ...
|
|
totalMessagesOnWeekday.set(weekday, totalMessagesOnWeekday.has(weekday) ? totalMessagesOnWeekday.get(weekday) + msgs : msgs);
|
|
};
|
|
|
|
for (let m = moment(from); m.diff(to, 'days') <= 0; m.add(1, 'days')) {
|
|
const date = {
|
|
gte: m,
|
|
lt: moment(m).add(1, 'days'),
|
|
};
|
|
|
|
const result = LivechatRooms.getAnalyticsMetricsBetweenDate('l', date);
|
|
totalConversations += result.count();
|
|
|
|
result.forEach(summarize(m));
|
|
}
|
|
|
|
const busiestDay = this.getKeyHavingMaxValue(totalMessagesOnWeekday, '-'); // returns key with max value
|
|
|
|
// iterate through all busiestDay in given date-range and find busiest hour
|
|
for (let m = moment(from).day(busiestDay); m <= to; m.add(7, 'days')) {
|
|
if (m < from) { continue; }
|
|
|
|
for (let h = moment(m); h.diff(m, 'days') <= 0; h.add(1, 'hours')) {
|
|
const date = {
|
|
gte: h,
|
|
lt: moment(h).add(1, 'hours'),
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
msgs,
|
|
}) => {
|
|
const dayHour = h.format('H'); // @int : 0, 1, ... 23
|
|
totalMessagesInHour.set(dayHour, totalMessagesInHour.has(dayHour) ? totalMessagesInHour.get(dayHour) + msgs : msgs);
|
|
});
|
|
}
|
|
}
|
|
|
|
const busiestHour = this.getKeyHavingMaxValue(totalMessagesInHour, -1);
|
|
|
|
const data = [{
|
|
title: 'Total_conversations',
|
|
value: totalConversations,
|
|
}, {
|
|
title: 'Open_conversations',
|
|
value: openConversations,
|
|
}, {
|
|
title: 'Total_messages',
|
|
value: totalMessages,
|
|
}, {
|
|
title: 'Busiest_day',
|
|
value: busiestDay,
|
|
}, {
|
|
title: 'Conversations_per_day',
|
|
value: (totalConversations / days).toFixed(2),
|
|
}, {
|
|
title: 'Busiest_time',
|
|
value: busiestHour > 0 ? `${ moment(busiestHour, ['H']).format('hA') }-${ moment((parseInt(busiestHour) + 1) % 24, ['H']).format('hA') }` : '-',
|
|
}];
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array[Object]}
|
|
*/
|
|
Productivity(from, to) {
|
|
let avgResponseTime = 0;
|
|
let firstResponseTime = 0;
|
|
let avgReactionTime = 0;
|
|
let count = 0;
|
|
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
}) => {
|
|
if (metrics && metrics.response && metrics.reaction) {
|
|
avgResponseTime += metrics.response.avg;
|
|
firstResponseTime += metrics.response.ft;
|
|
avgReactionTime += metrics.reaction.ft;
|
|
count++;
|
|
}
|
|
});
|
|
|
|
if (count) {
|
|
avgResponseTime /= count;
|
|
firstResponseTime /= count;
|
|
avgReactionTime /= count;
|
|
}
|
|
|
|
const data = [{
|
|
title: 'Avg_response_time',
|
|
value: secondsToHHMMSS(avgResponseTime.toFixed(2)),
|
|
}, {
|
|
title: 'Avg_first_response_time',
|
|
value: secondsToHHMMSS(firstResponseTime.toFixed(2)),
|
|
}, {
|
|
title: 'Avg_reaction_time',
|
|
value: secondsToHHMMSS(avgReactionTime.toFixed(2)),
|
|
}];
|
|
|
|
return data;
|
|
},
|
|
},
|
|
|
|
AgentOverviewData: {
|
|
/**
|
|
* do operation equivalent to map[key] += value
|
|
*
|
|
*/
|
|
updateMap(map, key, value) {
|
|
map.set(key, map.has(key) ? map.get(key) + value : value);
|
|
},
|
|
|
|
/**
|
|
* Sort array of objects by value property of object
|
|
* @param {Array(Object)} data
|
|
* @param {Boolean} [inv=false] reverse sort
|
|
*/
|
|
sortByValue(data, inv = false) {
|
|
data.sort(function(a, b) { // sort array
|
|
if (parseFloat(a.value) > parseFloat(b.value)) {
|
|
return inv ? -1 : 1; // if inv, reverse sort
|
|
}
|
|
if (parseFloat(a.value) < parseFloat(b.value)) {
|
|
return inv ? 1 : -1;
|
|
}
|
|
return 0;
|
|
});
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Total_conversations(from, to) {
|
|
let total = 0;
|
|
const agentConversations = new Map(); // stores total conversations for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: '%_of_conversations',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy) {
|
|
this.updateMap(agentConversations, servedBy.username, 1);
|
|
total++;
|
|
}
|
|
});
|
|
|
|
agentConversations.forEach((value, key) => { // calculate percentage
|
|
const percentage = (value / total * 100).toFixed(2);
|
|
|
|
data.data.push({
|
|
name: key,
|
|
value: percentage,
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, true); // reverse sort array
|
|
|
|
data.data.forEach((value) => {
|
|
value.value = `${ value.value }%`;
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Avg_chat_duration(from, to) {
|
|
const agentChatDurations = new Map(); // stores total conversations for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Avg_chat_duration',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy && metrics && metrics.chatDuration) {
|
|
if (agentChatDurations.has(servedBy.username)) {
|
|
agentChatDurations.set(servedBy.username, {
|
|
chatDuration: agentChatDurations.get(servedBy.username).chatDuration + metrics.chatDuration,
|
|
total: agentChatDurations.get(servedBy.username).total + 1,
|
|
});
|
|
} else {
|
|
agentChatDurations.set(servedBy.username, {
|
|
chatDuration: metrics.chatDuration,
|
|
total: 1,
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
agentChatDurations.forEach((obj, key) => { // calculate percentage
|
|
const avg = (obj.chatDuration / obj.total).toFixed(2);
|
|
|
|
data.data.push({
|
|
name: key,
|
|
value: avg,
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, true); // reverse sort array
|
|
|
|
data.data.forEach((obj) => {
|
|
obj.value = secondsToHHMMSS(obj.value);
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Total_messages(from, to) {
|
|
const agentMessages = new Map(); // stores total conversations for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Total_messages',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
servedBy,
|
|
msgs,
|
|
}) => {
|
|
if (servedBy) {
|
|
this.updateMap(agentMessages, servedBy.username, msgs);
|
|
}
|
|
});
|
|
|
|
agentMessages.forEach((value, key) => { // calculate percentage
|
|
data.data.push({
|
|
name: key,
|
|
value,
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, true); // reverse sort array
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Avg_first_response_time(from, to) {
|
|
const agentAvgRespTime = new Map(); // stores avg response time for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Avg_first_response_time',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy && metrics && metrics.response && metrics.response.ft) {
|
|
if (agentAvgRespTime.has(servedBy.username)) {
|
|
agentAvgRespTime.set(servedBy.username, {
|
|
frt: agentAvgRespTime.get(servedBy.username).frt + metrics.response.ft,
|
|
total: agentAvgRespTime.get(servedBy.username).total + 1,
|
|
});
|
|
} else {
|
|
agentAvgRespTime.set(servedBy.username, {
|
|
frt: metrics.response.ft,
|
|
total: 1,
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
agentAvgRespTime.forEach((obj, key) => { // calculate avg
|
|
const avg = obj.frt / obj.total;
|
|
|
|
data.data.push({
|
|
name: key,
|
|
value: avg.toFixed(2),
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, false); // sort array
|
|
|
|
data.data.forEach((obj) => {
|
|
obj.value = secondsToHHMMSS(obj.value);
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Best_first_response_time(from, to) {
|
|
const agentFirstRespTime = new Map(); // stores avg response time for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Best_first_response_time',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy && metrics && metrics.response && metrics.response.ft) {
|
|
if (agentFirstRespTime.has(servedBy.username)) {
|
|
agentFirstRespTime.set(servedBy.username, Math.min(agentFirstRespTime.get(servedBy.username), metrics.response.ft));
|
|
} else {
|
|
agentFirstRespTime.set(servedBy.username, metrics.response.ft);
|
|
}
|
|
}
|
|
});
|
|
|
|
agentFirstRespTime.forEach((value, key) => { // calculate avg
|
|
data.data.push({
|
|
name: key,
|
|
value: value.toFixed(2),
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, false); // sort array
|
|
|
|
data.data.forEach((obj) => {
|
|
obj.value = secondsToHHMMSS(obj.value);
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Avg_response_time(from, to) {
|
|
const agentAvgRespTime = new Map(); // stores avg response time for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Avg_response_time',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy && metrics && metrics.response && metrics.response.avg) {
|
|
if (agentAvgRespTime.has(servedBy.username)) {
|
|
agentAvgRespTime.set(servedBy.username, {
|
|
avg: agentAvgRespTime.get(servedBy.username).avg + metrics.response.avg,
|
|
total: agentAvgRespTime.get(servedBy.username).total + 1,
|
|
});
|
|
} else {
|
|
agentAvgRespTime.set(servedBy.username, {
|
|
avg: metrics.response.avg,
|
|
total: 1,
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
agentAvgRespTime.forEach((obj, key) => { // calculate avg
|
|
const avg = obj.avg / obj.total;
|
|
|
|
data.data.push({
|
|
name: key,
|
|
value: avg.toFixed(2),
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, false); // sort array
|
|
|
|
data.data.forEach((obj) => {
|
|
obj.value = secondsToHHMMSS(obj.value);
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
*
|
|
* @param {Date} from
|
|
* @param {Date} to
|
|
*
|
|
* @returns {Array(Object), Array(Object)}
|
|
*/
|
|
Avg_reaction_time(from, to) {
|
|
const agentAvgReactionTime = new Map(); // stores avg reaction time for each agent
|
|
const date = {
|
|
gte: from,
|
|
lt: to.add(1, 'days'),
|
|
};
|
|
|
|
const data = {
|
|
head: [{
|
|
name: 'Agent',
|
|
}, {
|
|
name: 'Avg_reaction_time',
|
|
}],
|
|
data: [],
|
|
};
|
|
|
|
LivechatRooms.getAnalyticsMetricsBetweenDate('l', date).forEach(({
|
|
metrics,
|
|
servedBy,
|
|
}) => {
|
|
if (servedBy && metrics && metrics.reaction && metrics.reaction.ft) {
|
|
if (agentAvgReactionTime.has(servedBy.username)) {
|
|
agentAvgReactionTime.set(servedBy.username, {
|
|
frt: agentAvgReactionTime.get(servedBy.username).frt + metrics.reaction.ft,
|
|
total: agentAvgReactionTime.get(servedBy.username).total + 1,
|
|
});
|
|
} else {
|
|
agentAvgReactionTime.set(servedBy.username, {
|
|
frt: metrics.reaction.ft,
|
|
total: 1,
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
agentAvgReactionTime.forEach((obj, key) => { // calculate avg
|
|
const avg = obj.frt / obj.total;
|
|
|
|
data.data.push({
|
|
name: key,
|
|
value: avg.toFixed(2),
|
|
});
|
|
});
|
|
|
|
this.sortByValue(data.data, false); // sort array
|
|
|
|
data.data.forEach((obj) => {
|
|
obj.value = secondsToHHMMSS(obj.value);
|
|
});
|
|
|
|
return data;
|
|
},
|
|
},
|
|
};
|
|
|