mirror of
https://github.com/go-vikunja/vikunja.git
synced 2026-05-01 01:16:40 +00:00
Also fix BasicAuth to check for status errors from checkUserCaldavTokens before falling through to password-based auth.
774 lines
30 KiB
Go
774 lines
30 KiB
Go
// Vikunja is a to-do list application to facilitate your life.
|
|
// Copyright 2018-present Vikunja and contributors. All rights reserved.
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
package webtests
|
|
|
|
import (
|
|
"encoding/xml"
|
|
"net/http"
|
|
"strings"
|
|
"testing"
|
|
|
|
"code.vikunja.io/api/pkg/db"
|
|
"code.vikunja.io/api/pkg/models"
|
|
"code.vikunja.io/api/pkg/routes/caldav"
|
|
|
|
ics "github.com/arran4/golang-ical"
|
|
"github.com/labstack/echo/v5"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestCaldav(t *testing.T) {
|
|
t.Run("Delivers VTODO for project", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.ProjectHandler, &testuser15, ``, nil, map[string]string{"project": "36"})
|
|
require.NoError(t, err)
|
|
assert.Contains(t, rec.Body.String(), "BEGIN:VCALENDAR")
|
|
assert.Contains(t, rec.Body.String(), "PRODID:-//Vikunja Todo App//EN")
|
|
assert.Contains(t, rec.Body.String(), "X-WR-CALNAME:Project 36 for Caldav tests")
|
|
assert.Contains(t, rec.Body.String(), "BEGIN:VTODO")
|
|
assert.Contains(t, rec.Body.String(), "END:VTODO")
|
|
assert.Contains(t, rec.Body.String(), "END:VCALENDAR")
|
|
})
|
|
t.Run("Import VTODO", func(t *testing.T) {
|
|
const vtodo = `BEGIN:VCALENDAR
|
|
VERSION:2.0
|
|
X-PUBLISHED-TTL:PT4H
|
|
X-WR-CALNAME:List 36 for Caldav tests
|
|
PRODID:-//Vikunja Todo App//EN
|
|
BEGIN:VTODO
|
|
UID:uid
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav Task 1
|
|
CATEGORIES:tag1,tag2,tag3
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
BEGIN:VALARM
|
|
TRIGGER;VALUE=DATE-TIME:20230304T150000Z
|
|
ACTION:DISPLAY
|
|
END:VALARM
|
|
END:VTODO
|
|
END:VCALENDAR`
|
|
|
|
e, _ := setupTestEnv()
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, vtodo, nil, map[string]string{"project": "36", "task": "uid"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
})
|
|
t.Run("Export VTODO", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test"})
|
|
require.NoError(t, err)
|
|
assert.Contains(t, rec.Body.String(), "BEGIN:VCALENDAR")
|
|
assert.Contains(t, rec.Body.String(), "SUMMARY:Title Caldav Test")
|
|
assert.Contains(t, rec.Body.String(), "DESCRIPTION:Description Caldav Test")
|
|
assert.Contains(t, rec.Body.String(), "DUE:20230301T150000Z")
|
|
assert.Contains(t, rec.Body.String(), "PRIORITY:3")
|
|
assert.Contains(t, rec.Body.String(), "CATEGORIES:Label #4")
|
|
assert.Contains(t, rec.Body.String(), "BEGIN:VALARM")
|
|
assert.Contains(t, rec.Body.String(), "TRIGGER;VALUE=DATE-TIME:20230304T150000Z")
|
|
assert.Contains(t, rec.Body.String(), "ACTION:DISPLAY")
|
|
assert.Contains(t, rec.Body.String(), "END:VALARM")
|
|
})
|
|
}
|
|
|
|
func TestCaldavDiscovery(t *testing.T) {
|
|
t.Run("Project home set depth 1 includes child projects but not itself", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
propfindBody := `<?xml version="1.0" encoding="utf-8" ?>
|
|
<A:propfind xmlns:A="DAV:" xmlns:B="urn:ietf:params:xml:ns:caldav">
|
|
<A:prop>
|
|
<A:current-user-principal />
|
|
<B:calendar-home-set />
|
|
<A:resourcetype />
|
|
</A:prop>
|
|
</A:propfind>`
|
|
|
|
c, rec := createRequest(e, "PROPFIND", propfindBody, nil, nil)
|
|
c.Request().Header.Set(echo.HeaderContentType, echo.MIMETextXML)
|
|
c.Request().Header.Set("Depth", "1")
|
|
c.Request().URL.Path = caldav.ProjectBasePath + "/"
|
|
c.Request().RequestURI = caldav.ProjectBasePath + "/"
|
|
|
|
result, _ := caldav.BasicAuth(c, testuser15.Username, "12345678")
|
|
require.True(t, result)
|
|
|
|
err := caldav.ProjectHandler(c)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 207, rec.Result().StatusCode)
|
|
|
|
responseBody := rec.Body.String()
|
|
assert.Contains(t, responseBody, "/dav/projects/36")
|
|
assert.NotContains(t, responseBody, "<d:href>/dav/projects/</d:href>")
|
|
assert.NotContains(t, responseBody, "<D:href>/dav/projects/</D:href>")
|
|
})
|
|
|
|
t.Run("Project home set depth 0 returns the home set itself", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
propfindBody := `<?xml version="1.0" encoding="utf-8" ?>
|
|
<A:propfind xmlns:A="DAV:" xmlns:B="urn:ietf:params:xml:ns:caldav">
|
|
<A:prop>
|
|
<A:current-user-principal />
|
|
<B:calendar-home-set />
|
|
<A:resourcetype />
|
|
</A:prop>
|
|
</A:propfind>`
|
|
|
|
c, rec := createRequest(e, "PROPFIND", propfindBody, nil, nil)
|
|
c.Request().Header.Set(echo.HeaderContentType, echo.MIMETextXML)
|
|
c.Request().Header.Set("Depth", "0")
|
|
c.Request().URL.Path = caldav.ProjectBasePath + "/"
|
|
c.Request().RequestURI = caldav.ProjectBasePath + "/"
|
|
|
|
result, _ := caldav.BasicAuth(c, testuser15.Username, "12345678")
|
|
require.True(t, result)
|
|
|
|
err := caldav.ProjectHandler(c)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 207, rec.Result().StatusCode)
|
|
|
|
responseBody := rec.Body.String()
|
|
assert.Contains(t, responseBody, "/dav/projects/")
|
|
})
|
|
|
|
t.Run("Principal discovery points to normalized project home set path", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
propfindBody := `<?xml version="1.0" encoding="utf-8" ?>
|
|
<A:propfind xmlns:A="DAV:" xmlns:B="urn:ietf:params:xml:ns:caldav">
|
|
<A:prop>
|
|
<A:current-user-principal />
|
|
<B:calendar-home-set />
|
|
</A:prop>
|
|
</A:propfind>`
|
|
|
|
c, rec := createRequest(e, "PROPFIND", propfindBody, nil, nil)
|
|
c.Request().Header.Set(echo.HeaderContentType, echo.MIMETextXML)
|
|
c.Request().URL.Path = caldav.PrincipalBasePath + "/user15/"
|
|
c.Request().RequestURI = caldav.PrincipalBasePath + "/user15/"
|
|
|
|
result, _ := caldav.BasicAuth(c, testuser15.Username, "12345678")
|
|
require.True(t, result)
|
|
|
|
err := caldav.PrincipalHandler(c)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 207, rec.Result().StatusCode)
|
|
|
|
responseBody := rec.Body.String()
|
|
assert.Contains(t, responseBody, "<D:href>/dav/projects</D:href>")
|
|
assert.NotContains(t, responseBody, "/dav//projects/")
|
|
})
|
|
}
|
|
|
|
func TestCaldavSubtasks(t *testing.T) {
|
|
const vtodoHeader = `BEGIN:VCALENDAR
|
|
VERSION:2.0
|
|
X-PUBLISHED-TTL:PT4H
|
|
X-WR-CALNAME:Project 36 for Caldav tests
|
|
PRODID:-//Vikunja Todo App//EN
|
|
`
|
|
const vtodoFooter = `
|
|
END:VCALENDAR`
|
|
|
|
t.Run("Import Task & Subtask", func(t *testing.T) {
|
|
|
|
const vtodoParentTaskStub = `BEGIN:VTODO
|
|
UID:uid_parent_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav parent task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
END:VTODO`
|
|
|
|
const vtodoChildTaskStub = `BEGIN:VTODO
|
|
UID:uid_child_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav child task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_parent_import
|
|
END:VTODO`
|
|
|
|
const vtodoGrandChildTaskStub = `
|
|
BEGIN:VTODO
|
|
UID:uid_grand_child_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav grand child task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_child_import
|
|
END:VTODO`
|
|
|
|
e, _ := setupTestEnv()
|
|
|
|
const parentVTODO = vtodoHeader + vtodoParentTaskStub + vtodoFooter
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentVTODO, nil, map[string]string{"project": "36", "task": "uid_parent_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
const childVTODO = vtodoHeader + vtodoChildTaskStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, childVTODO, nil, map[string]string{"project": "36", "task": "uid_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
const grandChildVTODO = vtodoHeader + vtodoGrandChildTaskStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, grandChildVTODO, nil, map[string]string{"project": "36", "task": "uid_grand_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.ProjectHandler, &testuser15, ``, nil, map[string]string{"project": "36"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_parent_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid_child_import")
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_child_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid_parent_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid_grand_child_import")
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_grand_child_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid_child_import")
|
|
})
|
|
|
|
t.Run("Import Task & Subtask (Reverse - Subtask first)", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
const vtodoGrandChildTaskStub = `
|
|
BEGIN:VTODO
|
|
UID:uid_grand_child_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav grand child task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_child_import
|
|
END:VTODO`
|
|
|
|
const grandChildVTODO = vtodoHeader + vtodoGrandChildTaskStub + vtodoFooter
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, grandChildVTODO, nil, map[string]string{"project": "36", "task": "uid_grand_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
const vtodoChildTaskStub = `BEGIN:VTODO
|
|
UID:uid_child_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav child task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_parent_import
|
|
RELATED-TO;RELTYPE=CHILD:uid_grand_child_import
|
|
END:VTODO`
|
|
|
|
const childVTODO = vtodoHeader + vtodoChildTaskStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, childVTODO, nil, map[string]string{"project": "36", "task": "uid_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
const vtodoParentTaskStub = `BEGIN:VTODO
|
|
UID:uid_parent_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav parent task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=CHILD:uid_child_import
|
|
END:VTODO`
|
|
|
|
const parentVTODO = vtodoHeader + vtodoParentTaskStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentVTODO, nil, map[string]string{"project": "36", "task": "uid_parent_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.ProjectHandler, &testuser15, ``, nil, map[string]string{"project": "36"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_parent_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid_child_import")
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_child_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid_parent_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid_grand_child_import")
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_grand_child_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid_child_import")
|
|
})
|
|
|
|
t.Run("Import Task & Subtask (Reverse - Parent without RELATED-TO)", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
// Step 1: Subtask arrives FIRST, referencing a parent that doesn't exist yet.
|
|
// This is the standard Tasks.org behavior: only the child has RELATED-TO.
|
|
const vtodoSubtaskStub = `BEGIN:VTODO
|
|
UID:uid_child_no_reltype
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Subtask without parent RELTYPE
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_parent_no_reltype
|
|
END:VTODO`
|
|
|
|
const subtaskVTODO = vtodoHeader + vtodoSubtaskStub + vtodoFooter
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, subtaskVTODO, nil, map[string]string{"project": "36", "task": "uid_child_no_reltype"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 2: Parent arrives with NO RELATED-TO at all.
|
|
// This is how Tasks.org sends parent tasks — no RELATED-TO;RELTYPE=CHILD.
|
|
const vtodoParentStub = `BEGIN:VTODO
|
|
UID:uid_parent_no_reltype
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Parent without RELTYPE
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
END:VTODO`
|
|
|
|
const parentVTODO = vtodoHeader + vtodoParentStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentVTODO, nil, map[string]string{"project": "36", "task": "uid_parent_no_reltype"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 3: Verify relations at the DB level.
|
|
s := db.NewSession()
|
|
defer s.Close()
|
|
|
|
childTasks, err := models.GetTasksByUIDs(s, []string{"uid_child_no_reltype"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, childTasks, 1)
|
|
childTask := childTasks[0]
|
|
|
|
parentTasks, err := models.GetTasksByUIDs(s, []string{"uid_parent_no_reltype"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, parentTasks, 1)
|
|
parentTask := parentTasks[0]
|
|
|
|
// Parent should have correct title (DUMMY should have been replaced)
|
|
assert.Equal(t, "Parent without RELTYPE", parentTask.Title)
|
|
|
|
// No DUMMY-UID tasks should remain
|
|
db.AssertMissing(t, "tasks", map[string]interface{}{
|
|
"title": "DUMMY-UID-uid_parent_no_reltype",
|
|
})
|
|
|
|
// Subtask should still have parenttask relation to parent
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": childTask.ID,
|
|
"other_task_id": parentTask.ID,
|
|
"relation_kind": models.RelationKindParenttask,
|
|
}, false)
|
|
|
|
// Parent should have the inverse subtask relation
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": parentTask.ID,
|
|
"other_task_id": childTask.ID,
|
|
"relation_kind": models.RelationKindSubtask,
|
|
}, false)
|
|
})
|
|
|
|
t.Run("Parent re-sync without RELATED-TO preserves child relations", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
// Step 1: Parent created first (no RELATED-TO).
|
|
const vtodoParentStub = `BEGIN:VTODO
|
|
UID:uid_parent_resync
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Parent for resync test
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
END:VTODO`
|
|
|
|
const parentVTODO = vtodoHeader + vtodoParentStub + vtodoFooter
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentVTODO, nil, map[string]string{"project": "36", "task": "uid_parent_resync"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 2: Subtask arrives with RELATED-TO;RELTYPE=PARENT.
|
|
const vtodoSubtaskStub = `BEGIN:VTODO
|
|
UID:uid_child_resync
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Child for resync test
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_parent_resync
|
|
END:VTODO`
|
|
|
|
const subtaskVTODO = vtodoHeader + vtodoSubtaskStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, subtaskVTODO, nil, map[string]string{"project": "36", "task": "uid_child_resync"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 3: Parent is re-synced (updated) — still no RELATED-TO.
|
|
// This simulates DAVx5 re-syncing the parent after a change (e.g., title update).
|
|
const vtodoParentUpdatedStub = `BEGIN:VTODO
|
|
UID:uid_parent_resync
|
|
DTSTAMP:20230302T073337Z
|
|
SUMMARY:Parent for resync test (updated)
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230302T073337Z
|
|
END:VTODO`
|
|
|
|
const parentUpdatedVTODO = vtodoHeader + vtodoParentUpdatedStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentUpdatedVTODO, nil, map[string]string{"project": "36", "task": "uid_parent_resync"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 4: Verify relations still intact after parent re-sync.
|
|
s := db.NewSession()
|
|
defer s.Close()
|
|
|
|
parentTasks, err := models.GetTasksByUIDs(s, []string{"uid_parent_resync"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, parentTasks, 1)
|
|
parentTask := parentTasks[0]
|
|
|
|
childTasks, err := models.GetTasksByUIDs(s, []string{"uid_child_resync"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, childTasks, 1)
|
|
childTask := childTasks[0]
|
|
|
|
// Parent should have updated title
|
|
assert.Equal(t, "Parent for resync test (updated)", parentTask.Title)
|
|
|
|
// Child should still have parenttask relation to parent
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": childTask.ID,
|
|
"other_task_id": parentTask.ID,
|
|
"relation_kind": models.RelationKindParenttask,
|
|
}, false)
|
|
|
|
// Parent should still have inverse subtask relation
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": parentTask.ID,
|
|
"other_task_id": childTask.ID,
|
|
"relation_kind": models.RelationKindSubtask,
|
|
}, false)
|
|
})
|
|
|
|
t.Run("Multiple subtasks with same parent (one-sided RELATED-TO)", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
// Step 1: First subtask arrives, parent doesn't exist yet.
|
|
const vtodoSubtask1Stub = `BEGIN:VTODO
|
|
UID:uid_multi_child_1
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Multi child 1
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_multi_parent
|
|
END:VTODO`
|
|
|
|
const subtask1VTODO = vtodoHeader + vtodoSubtask1Stub + vtodoFooter
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, subtask1VTODO, nil, map[string]string{"project": "36", "task": "uid_multi_child_1"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 2: Second subtask arrives, parent should exist as DUMMY now.
|
|
const vtodoSubtask2Stub = `BEGIN:VTODO
|
|
UID:uid_multi_child_2
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Multi child 2
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_multi_parent
|
|
END:VTODO`
|
|
|
|
const subtask2VTODO = vtodoHeader + vtodoSubtask2Stub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, subtask2VTODO, nil, map[string]string{"project": "36", "task": "uid_multi_child_2"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 3: Parent arrives with NO RELATED-TO.
|
|
const vtodoParentStub = `BEGIN:VTODO
|
|
UID:uid_multi_parent
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Multi parent
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
END:VTODO`
|
|
|
|
const parentVTODO = vtodoHeader + vtodoParentStub + vtodoFooter
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, parentVTODO, nil, map[string]string{"project": "36", "task": "uid_multi_parent"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
// Step 4: Verify all relations intact and no DUMMY tasks.
|
|
s := db.NewSession()
|
|
defer s.Close()
|
|
|
|
parentTasks, err := models.GetTasksByUIDs(s, []string{"uid_multi_parent"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, parentTasks, 1)
|
|
parentTask := parentTasks[0]
|
|
|
|
child1Tasks, err := models.GetTasksByUIDs(s, []string{"uid_multi_child_1"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, child1Tasks, 1)
|
|
child1Task := child1Tasks[0]
|
|
|
|
child2Tasks, err := models.GetTasksByUIDs(s, []string{"uid_multi_child_2"}, &testuser15)
|
|
require.NoError(t, err)
|
|
require.Len(t, child2Tasks, 1)
|
|
child2Task := child2Tasks[0]
|
|
|
|
// Parent should have correct title
|
|
assert.Equal(t, "Multi parent", parentTask.Title)
|
|
|
|
// No DUMMY-UID tasks should remain
|
|
db.AssertMissing(t, "tasks", map[string]interface{}{
|
|
"title": "DUMMY-UID-uid_multi_parent",
|
|
})
|
|
|
|
// Child 1 should have parenttask relation to parent
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": child1Task.ID,
|
|
"other_task_id": parentTask.ID,
|
|
"relation_kind": models.RelationKindParenttask,
|
|
}, false)
|
|
|
|
// Child 2 should have parenttask relation to parent
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": child2Task.ID,
|
|
"other_task_id": parentTask.ID,
|
|
"relation_kind": models.RelationKindParenttask,
|
|
}, false)
|
|
|
|
// Parent should have inverse subtask relations to both children
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": parentTask.ID,
|
|
"other_task_id": child1Task.ID,
|
|
"relation_kind": models.RelationKindSubtask,
|
|
}, false)
|
|
db.AssertExists(t, "task_relations", map[string]interface{}{
|
|
"task_id": parentTask.ID,
|
|
"other_task_id": child2Task.ID,
|
|
"relation_kind": models.RelationKindSubtask,
|
|
}, false)
|
|
})
|
|
|
|
t.Run("Delete Subtask", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodDelete, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-child-task"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 204, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodDelete, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-child-task-2"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 204, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-parent-task"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
|
|
assert.NotContains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid-caldav-test-child-task")
|
|
assert.NotContains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid-caldav-test-child-task-2")
|
|
})
|
|
|
|
t.Run("Delete Parent Task", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodDelete, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-parent-task"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 204, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-child-task"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
|
|
assert.NotContains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid-caldav-test-parent-task")
|
|
})
|
|
|
|
}
|
|
|
|
func TestCaldavSubtasksDifferentLists(t *testing.T) {
|
|
t.Run("Import Parent Task & Child Task Different Lists", func(t *testing.T) {
|
|
const vtodoParentTask = `BEGIN:VCALENDAR
|
|
VERSION:2.0
|
|
X-PUBLISHED-TTL:PT4H
|
|
X-WR-CALNAME:Project 36 for Caldav tests
|
|
PRODID:-//Vikunja Todo App//EN
|
|
BEGIN:VTODO
|
|
UID:uid_parent_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav parent task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
END:VTODO
|
|
END:VCALENDAR`
|
|
|
|
const vtodoChildTask = `BEGIN:VCALENDAR
|
|
VERSION:2.0
|
|
X-PUBLISHED-TTL:PT4H
|
|
X-WR-CALNAME:Project 38 for Caldav tests
|
|
PRODID:-//Vikunja Todo App//EN
|
|
BEGIN:VTODO
|
|
UID:uid_child_import
|
|
DTSTAMP:20230301T073337Z
|
|
SUMMARY:Caldav child task
|
|
CREATED:20230301T073337Z
|
|
LAST-MODIFIED:20230301T073337Z
|
|
RELATED-TO;RELTYPE=PARENT:uid_parent_import
|
|
END:VTODO
|
|
END:VCALENDAR`
|
|
|
|
e, _ := setupTestEnv()
|
|
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, vtodoParentTask, nil, map[string]string{"project": "36", "task": "uid_parent_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodPut, caldav.TaskHandler, &testuser15, vtodoChildTask, nil, map[string]string{"project": "38", "task": "uid_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 201, rec.Result().StatusCode)
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid_parent_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_parent_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid_child_import")
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "38", "task": "uid_child_import"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
assert.Contains(t, rec.Body.String(), "UID:uid_child_import")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid_parent_import")
|
|
})
|
|
|
|
t.Run("Check relationships across lists", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
rec, err := newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "36", "task": "uid-caldav-test-parent-task-another-list"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
assert.Contains(t, rec.Body.String(), "UID:uid-caldav-test-parent-task-another-list")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=CHILD:uid-caldav-test-child-task-another-list")
|
|
|
|
rec, err = newCaldavTestRequestWithUser(t, e, http.MethodGet, caldav.TaskHandler, &testuser15, ``, nil, map[string]string{"project": "38", "task": "uid-caldav-test-child-task-another-list"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 200, rec.Result().StatusCode)
|
|
assert.Contains(t, rec.Body.String(), "UID:uid-caldav-test-child-task-another-list")
|
|
assert.Contains(t, rec.Body.String(), "RELATED-TO;RELTYPE=PARENT:uid-caldav-test-parent-task-another-list")
|
|
})
|
|
}
|
|
|
|
func TestCaldavProjectReport(t *testing.T) {
|
|
t.Run("REPORT calendar-query returns all tasks", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
|
|
// CalDAV REPORT request for calendar-query
|
|
reportBody := `<?xml version="1.0" encoding="utf-8" ?>
|
|
<C:calendar-query xmlns:C="urn:ietf:params:xml:ns:caldav">
|
|
<D:prop xmlns:D="DAV:">
|
|
<D:getetag/>
|
|
<C:calendar-data/>
|
|
</D:prop>
|
|
<C:filter>
|
|
<C:comp-filter name="VCALENDAR">
|
|
<C:comp-filter name="VTODO"/>
|
|
</C:comp-filter>
|
|
</C:filter>
|
|
</C:calendar-query>`
|
|
|
|
rec, err := newCaldavTestRequestWithUser(t, e, "REPORT", caldav.ProjectHandler, &testuser15, reportBody, nil, map[string]string{"project": "36"})
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 207, rec.Result().StatusCode) // Multi-Status response
|
|
|
|
responseBody := rec.Body.String()
|
|
|
|
assert.Contains(t, responseBody, "multistatus")
|
|
assert.Contains(t, responseBody, "response")
|
|
assert.Contains(t, responseBody, "href")
|
|
assert.Contains(t, responseBody, "propstat")
|
|
|
|
// Parse XML to verify structure
|
|
type Multistatus struct {
|
|
Response []struct {
|
|
Href string `xml:"href"`
|
|
Propstat struct {
|
|
Prop struct {
|
|
Getetag string `xml:"getetag"`
|
|
CalendarData string `xml:"calendar-data"`
|
|
} `xml:"prop"`
|
|
} `xml:"propstat"`
|
|
} `xml:"response"`
|
|
}
|
|
|
|
var multistatus Multistatus
|
|
err = xml.Unmarshal([]byte(responseBody), &multistatus)
|
|
require.NoError(t, err)
|
|
|
|
assert.Len(t, multistatus.Response, 5, "Should have all tasks from the project")
|
|
|
|
for i, response := range multistatus.Response {
|
|
assert.NotEmpty(t, response.Href, "Response %d should have an href", i)
|
|
assert.NotEmpty(t, response.Propstat.Prop.CalendarData, "Response %d should have calendar-data", i)
|
|
assert.NotEmpty(t, response.Propstat.Prop.Getetag, "Response %d should have an ETag", i)
|
|
|
|
calendarData := response.Propstat.Prop.CalendarData
|
|
|
|
cal, err := ics.ParseCalendar(strings.NewReader(calendarData))
|
|
require.NoError(t, err, "Response %d should contain valid iCalendar data", i)
|
|
|
|
require.Len(t, cal.Components, 1, "Response %d should contain exactly one VTODO component")
|
|
|
|
component := cal.Components[0]
|
|
require.IsType(t, &ics.VTodo{}, component)
|
|
|
|
vtodo, _ := component.(*ics.VTodo)
|
|
uid := vtodo.GetProperty(ics.ComponentPropertyUniqueId)
|
|
assert.NotEmpty(t, uid.Value, "Response %d VTODO UID should not be empty", i)
|
|
|
|
summary := vtodo.GetProperty(ics.ComponentPropertySummary)
|
|
assert.NotEmpty(t, summary.Value, "Response %d VTODO SUMMARY should not be empty", i)
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestCaldavTOTPBlocksBasicAuth(t *testing.T) {
|
|
t.Run("Basic auth with password is rejected when TOTP is enabled", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
c, _ := createRequest(e, http.MethodGet, "", nil, nil)
|
|
|
|
// testuser10 has TOTP enabled via fixtures.
|
|
// "12345678" is the plaintext password for all test users.
|
|
result, err := caldav.BasicAuth(c, testuser10.Username, "12345678")
|
|
require.NoError(t, err)
|
|
assert.False(t, result, "BasicAuth should reject password login when user has TOTP enabled")
|
|
})
|
|
|
|
t.Run("Basic auth with caldav token still works when TOTP is enabled", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
c, _ := createRequest(e, http.MethodGet, "", nil, nil)
|
|
|
|
// testuser10 has TOTP enabled AND a CalDAV token (kind=4) in fixtures.
|
|
// "caldavtesttoken" is the plaintext of the bcrypt hash in user_tokens.yml.
|
|
// CalDAV token auth should bypass the TOTP check.
|
|
result, err := caldav.BasicAuth(c, testuser10.Username, "caldavtesttoken")
|
|
require.NoError(t, err)
|
|
assert.True(t, result, "BasicAuth with CalDAV token should succeed even when TOTP is enabled")
|
|
})
|
|
}
|
|
|
|
func TestCaldavDisabledUserRejected(t *testing.T) {
|
|
t.Run("disabled user cannot authenticate via CalDAV", func(t *testing.T) {
|
|
e, _ := setupTestEnv()
|
|
c, _ := createRequest(e, http.MethodGet, "", nil, nil)
|
|
|
|
// user17 is disabled (status=2), password is "12345678"
|
|
result, err := caldav.BasicAuth(c, "user17", "12345678")
|
|
require.NoError(t, err)
|
|
assert.False(t, result, "disabled user should not be able to authenticate via CalDAV")
|
|
})
|
|
}
|