* Add next suggestions and remove console logs * Add pomodoro timer * Add pomodoro switch in settings * Fix pomodoro setting * Add timezones to settings * Fix an issue with password reset * Cleanup * Sort tags alphabetically * Clean up today's view * Add an indicator for repeatedly added to today * Refactor tags * Add due date today item * Move recurrence to the subtitle area * Fix today layout * Add a badge to Inbox items * Move inbox badge to sidebar * Add quotes and progress bar * Add translations for quotes * Fix test issues * Add helper script for docker local * Set up overdue tasks * Add linux/arm/v7 build to deploy script * Add linux/arm/v7 build to deploy script pt2 * Fix an issue with helmet and SSL * Add volume db persistence * Fix cog icon issues
153 lines
No EOL
4.8 KiB
JavaScript
153 lines
No EOL
4.8 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; |