Files
vikunja/frontend/cypress/e2e/task/overview.spec.ts
Claude Loop aeb6a57e7a fix: resolve API intercept alias conflicts in task/overview.spec.ts
- Fix multiple cy.intercept() calls with same '@loadTasks' alias
- Replace with single regex pattern to match all task loading endpoints
- Prevents Cypress from using only the last defined intercept
- Should resolve timeouts in overview task visibility tests

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-09-21 06:55:06 +00:00

197 lines
6.0 KiB
TypeScript

import {createFakeUserAndLogin} from '../../support/authenticateUser'
import {ProjectFactory} from '../../factories/project'
import {seed} from '../../support/seed'
import {TaskFactory} from '../../factories/task'
import {BucketFactory} from '../../factories/bucket'
import {updateUserSettings} from '../../support/updateUserSettings'
import {createDefaultViews} from "../project/prepareProjects";
function seedTasks(numberOfTasks = 50, startDueDate = new Date()) {
const project = ProjectFactory.create()[0]
const views = createDefaultViews(project.id)
BucketFactory.create(1, {
project_view_id: views[3].id,
})
const tasks = []
let dueDate = startDueDate
for (let i = 0; i < numberOfTasks; i++) {
const now = new Date()
dueDate = new Date(new Date(dueDate).setDate(dueDate.getDate() + 2))
tasks.push({
id: i + 1,
project_id: project.id,
done: false,
created_by_id: 1,
title: 'Test Task ' + i,
index: i + 1,
due_date: dueDate.toISOString(),
created: now.toISOString(),
updated: now.toISOString(),
})
}
seed(TaskFactory.table, tasks)
return {tasks, project}
}
describe('Home Page Task Overview', () => {
createFakeUserAndLogin()
beforeEach(() => {
TaskFactory.truncate()
ProjectFactory.truncate()
BucketFactory.truncate()
})
it('Should show tasks with a near due date first on the home page overview', () => {
const taskCount = 50
const {tasks} = seedTasks(taskCount)
cy.visit('/')
cy.get('[data-cy="showTasks"] .card')
.should('exist')
cy.get('[data-cy="showTasks"] .card .task')
.should('have.length.greaterThan', 0)
.each(([task], index) => {
expect(task.innerText).to.contain(tasks[index].title)
})
})
it('Should show overdue tasks first, then show other tasks', () => {
const now = new Date()
const oldDate = new Date(new Date(now).setDate(now.getDate() - 14))
const taskCount = 50
const {tasks} = seedTasks(taskCount, oldDate)
cy.visit('/')
cy.get('[data-cy="showTasks"] .card')
.should('exist')
cy.get('[data-cy="showTasks"] .card .task')
.should('have.length.greaterThan', 0)
.each(([task], index) => {
expect(task.innerText).to.contain(tasks[index].title)
})
})
it('Should show a new task with a very soon due date at the top', () => {
const {tasks, project} = seedTasks(49)
const newTaskTitle = 'New Task'
cy.visit('/')
TaskFactory.create(1, {
id: 999,
title: newTaskTitle,
project_id: tasks[0].project_id,
due_date: new Date().toISOString(),
}, false)
// Set up intercept before any navigation that might trigger API calls
// Set up comprehensive API intercept for all possible task loading endpoints
cy.intercept('GET', /\/api\/v1\/(projects\/\d+(\/views\/\d+)?\/tasks|tasks\/all)/).as('loadTasks')
cy.intercept('GET', '**/api/v1/projects/*').as('loadProject')
// Visit the project page first and wait for it to load
cy.visit(`/projects/${project.id}`)
cy.url().should('contain', `/projects/${project.id}/1`)
// Wait for project to load first, then tasks
cy.wait('@loadProject', { timeout: 30000 })
cy.wait('@loadTasks', {timeout: 30000})
cy.get('.tasks')
.should('exist')
cy.get('.tasks .task')
.should('contain.text', newTaskTitle)
cy.visit('/')
cy.get('[data-cy="showTasks"] .card .task')
.first()
.should('contain.text', newTaskTitle)
})
it('Should not show a new task without a date at the bottom when there are > 50 tasks', () => {
// We're not using the api here to create the task in order to verify the flow
const {tasks, project} = seedTasks(100)
const newTaskTitle = 'New Task'
cy.visit('/')
// Set up intercepts before navigation
// Set up comprehensive API intercept for all possible task loading endpoints
cy.intercept('GET', /\/api\/v1\/(projects\/\d+(\/views\/\d+)?\/tasks|tasks\/all)/).as('loadTasks')
cy.intercept('PUT', `**/api/v1/projects/${project.id}/views/*/tasks`).as('createTask')
cy.intercept('GET', '**/api/v1/projects/*').as('loadProject')
// Visit the project page and wait for it to load
cy.visit(`/projects/${project.id}`)
cy.url().should('contain', `/projects/${project.id}/1`)
// Wait for project to load first, then tasks
cy.wait('@loadProject', { timeout: 30000 })
cy.wait('@loadTasks', {timeout: 30000})
cy.get('.task-add textarea')
.should('be.visible')
.type(newTaskTitle+'{enter}')
// Wait for task creation to complete with shorter timeout to prevent hangs
cy.wait('@createTask', { timeout: 30000 })
cy.get('.tasks .task').should('contain.text', newTaskTitle)
cy.visit('/')
cy.get('[data-cy="showTasks"] .card .task')
.last()
.should('not.contain.text', newTaskTitle)
})
it('Should show a new task without a date at the bottom when there are < 50 tasks', () => {
const {tasks} = seedTasks(40)
const newTaskTitle = 'New Task'
TaskFactory.create(1, {
id: 999,
title: newTaskTitle,
project_id: tasks[0].project_id,
}, false)
cy.visit('/')
cy.get('[data-cy="showTasks"] .card .task')
.last()
.should('contain.text', newTaskTitle)
})
it('Should show a task without a due date added via default project at the bottom', () => {
const {project} = seedTasks(40)
updateUserSettings({
default_project_id: project.id,
overdue_tasks_reminders_time: '9:00',
})
const newTaskTitle = 'New Task'
cy.visit('/')
cy.get('.add-task-textarea')
.type(`${newTaskTitle}{enter}`)
cy.get('[data-cy="showTasks"] .card .task')
.last()
.should('contain.text', newTaskTitle)
})
it('Should show the cta buttons for new project when there are no tasks', () => {
cy.visit('/')
cy.get('.home.app-content .content')
.should('contain.text', 'Import your projects and tasks from other services into Vikunja:')
})
it('Should not show the cta buttons for new project when there are tasks', () => {
seedTasks()
cy.visit('/')
cy.get('.home.app-content .content')
.should('not.contain.text', 'You can create a new project for your new tasks:')
.should('not.contain.text', 'Or import your projects and tasks from other services into Vikunja:')
})
})