170 lines
5.6 KiB
JavaScript
170 lines
5.6 KiB
JavaScript
const express = require('express');
|
|
const { TaskEvent } = require('../models');
|
|
const TaskEventService = require('../services/taskEventService');
|
|
const router = express.Router();
|
|
|
|
// GET /api/task/:id/timeline - Get task event timeline
|
|
router.get('/task/:id/timeline', async (req, res) => {
|
|
try {
|
|
const timeline = await TaskEventService.getTaskTimeline(req.params.id);
|
|
|
|
// Filter to only show events for tasks owned by the current user
|
|
const userTimeline = timeline.filter(
|
|
(event) => event.user_id === req.currentUser.id
|
|
);
|
|
|
|
res.json(userTimeline);
|
|
} catch (error) {
|
|
console.error('Error fetching task timeline:', error);
|
|
res.status(500).json({ error: 'Failed to fetch task timeline' });
|
|
}
|
|
});
|
|
|
|
// GET /api/task/:id/completion-time - Get task completion analytics
|
|
router.get('/task/:id/completion-time', async (req, res) => {
|
|
try {
|
|
const completionTime = await TaskEventService.getTaskCompletionTime(
|
|
req.params.id
|
|
);
|
|
|
|
if (!completionTime) {
|
|
return res
|
|
.status(404)
|
|
.json({ error: 'Task completion data not found' });
|
|
}
|
|
|
|
res.json(completionTime);
|
|
} catch (error) {
|
|
console.error('Error fetching task completion time:', error);
|
|
res.status(500).json({ error: 'Failed to fetch task completion time' });
|
|
}
|
|
});
|
|
|
|
// GET /api/user/productivity-metrics - Get user productivity metrics
|
|
router.get('/user/productivity-metrics', async (req, res) => {
|
|
try {
|
|
const { startDate, endDate } = req.query;
|
|
|
|
const metrics = await TaskEventService.getUserProductivityMetrics(
|
|
req.currentUser.id,
|
|
startDate ? new Date(startDate) : null,
|
|
endDate ? new Date(endDate) : null
|
|
);
|
|
|
|
res.json(metrics);
|
|
} catch (error) {
|
|
console.error('Error fetching productivity metrics:', error);
|
|
res.status(500).json({ error: 'Failed to fetch productivity metrics' });
|
|
}
|
|
});
|
|
|
|
// GET /api/user/activity-summary - Get task activity summary
|
|
router.get('/user/activity-summary', async (req, res) => {
|
|
try {
|
|
const { startDate, endDate } = req.query;
|
|
|
|
if (!startDate || !endDate) {
|
|
return res
|
|
.status(400)
|
|
.json({ error: 'startDate and endDate are required' });
|
|
}
|
|
|
|
const activitySummary = await TaskEventService.getTaskActivitySummary(
|
|
req.currentUser.id,
|
|
new Date(startDate),
|
|
new Date(endDate)
|
|
);
|
|
|
|
res.json(activitySummary);
|
|
} catch (error) {
|
|
console.error('Error fetching activity summary:', error);
|
|
res.status(500).json({ error: 'Failed to fetch activity summary' });
|
|
}
|
|
});
|
|
|
|
// GET /api/tasks/completion-analytics - Get completion time analytics for multiple tasks
|
|
router.get('/tasks/completion-analytics', async (req, res) => {
|
|
try {
|
|
const { limit = 50, offset = 0, projectId } = req.query;
|
|
|
|
// Get completed tasks for the user
|
|
const { Task, Project } = require('../models');
|
|
const { Op } = require('sequelize');
|
|
|
|
const whereClause = {
|
|
user_id: req.currentUser.id,
|
|
status: 2, // completed
|
|
};
|
|
|
|
if (projectId) {
|
|
whereClause.project_id = projectId;
|
|
}
|
|
|
|
const completedTasks = await Task.findAll({
|
|
where: whereClause,
|
|
include: [
|
|
{ model: Project, attributes: ['name'], required: false },
|
|
],
|
|
order: [['completed_at', 'DESC']],
|
|
limit: parseInt(limit),
|
|
offset: parseInt(offset),
|
|
});
|
|
|
|
// Get completion time analytics for each task
|
|
const analytics = [];
|
|
for (const task of completedTasks) {
|
|
const completionTime = await TaskEventService.getTaskCompletionTime(
|
|
task.id
|
|
);
|
|
if (completionTime) {
|
|
analytics.push({
|
|
task_id: task.id,
|
|
task_name: task.name,
|
|
project_name: task.Project?.name || null,
|
|
...completionTime,
|
|
});
|
|
}
|
|
}
|
|
|
|
// Calculate summary statistics
|
|
const summary = {
|
|
total_tasks: analytics.length,
|
|
average_completion_hours:
|
|
analytics.length > 0
|
|
? analytics.reduce((sum, a) => sum + a.duration_hours, 0) /
|
|
analytics.length
|
|
: 0,
|
|
median_completion_hours: 0,
|
|
fastest_completion:
|
|
analytics.length > 0
|
|
? Math.min(...analytics.map((a) => a.duration_hours))
|
|
: 0,
|
|
slowest_completion:
|
|
analytics.length > 0
|
|
? Math.max(...analytics.map((a) => a.duration_hours))
|
|
: 0,
|
|
};
|
|
|
|
// Calculate median
|
|
if (analytics.length > 0) {
|
|
const sorted = analytics
|
|
.map((a) => a.duration_hours)
|
|
.sort((a, b) => a - b);
|
|
const middle = Math.floor(sorted.length / 2);
|
|
summary.median_completion_hours =
|
|
sorted.length % 2 === 0
|
|
? (sorted[middle - 1] + sorted[middle]) / 2
|
|
: sorted[middle];
|
|
}
|
|
|
|
res.json({
|
|
tasks: analytics,
|
|
summary,
|
|
});
|
|
} catch (error) {
|
|
console.error('Error fetching completion analytics:', error);
|
|
res.status(500).json({ error: 'Failed to fetch completion analytics' });
|
|
}
|
|
});
|
|
|
|
module.exports = router;
|