// Vikunja is a to-do list application to facilitate your life. // Copyright 2018-2021 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 Licensee 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 Licensee for more details. // // You should have received a copy of the GNU Affero General Public Licensee // along with this program. If not, see . package models import ( "reflect" "testing" "code.vikunja.io/api/pkg/db" "code.vikunja.io/api/pkg/user" "github.com/stretchr/testify/assert" ) func TestList_CreateOrUpdate(t *testing.T) { usr := &user.User{ ID: 1, Username: "user1", Email: "user1@example.com", } t.Run("create", func(t *testing.T) { t.Run("normal", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ Title: "test", Description: "Lorem Ipsum", NamespaceID: 1, } err := list.Create(s, usr) assert.NoError(t, err) err = s.Commit() assert.NoError(t, err) db.AssertExists(t, "lists", map[string]interface{}{ "id": list.ID, "title": list.Title, "description": list.Description, "namespace_id": list.NamespaceID, }, false) }) t.Run("nonexistant namespace", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ Title: "test", Description: "Lorem Ipsum", NamespaceID: 999999, } err := list.Create(s, usr) assert.Error(t, err) assert.True(t, IsErrNamespaceDoesNotExist(err)) _ = s.Close() }) t.Run("nonexistant owner", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() usr := &user.User{ID: 9482385} list := List{ Title: "test", Description: "Lorem Ipsum", NamespaceID: 1, } err := list.Create(s, usr) assert.Error(t, err) assert.True(t, user.IsErrUserDoesNotExist(err)) _ = s.Close() }) t.Run("existing identifier", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ Title: "test", Description: "Lorem Ipsum", Identifier: "test1", NamespaceID: 1, } err := list.Create(s, usr) assert.Error(t, err) assert.True(t, IsErrListIdentifierIsNotUnique(err)) _ = s.Close() }) t.Run("non ascii characters", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ Title: "приффки фсем", Description: "Lorem Ipsum", NamespaceID: 1, } err := list.Create(s, usr) assert.NoError(t, err) err = s.Commit() assert.NoError(t, err) db.AssertExists(t, "lists", map[string]interface{}{ "id": list.ID, "title": list.Title, "description": list.Description, "namespace_id": list.NamespaceID, }, false) }) }) t.Run("update", func(t *testing.T) { t.Run("normal", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ ID: 1, Title: "test", Description: "Lorem Ipsum", NamespaceID: 1, } list.Description = "Lorem Ipsum dolor sit amet." err := list.Update(s, usr) assert.NoError(t, err) err = s.Commit() assert.NoError(t, err) db.AssertExists(t, "lists", map[string]interface{}{ "id": list.ID, "title": list.Title, "description": list.Description, "namespace_id": list.NamespaceID, }, false) }) t.Run("nonexistant", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ ID: 99999999, Title: "test", } err := list.Update(s, usr) assert.Error(t, err) assert.True(t, IsErrListDoesNotExist(err)) _ = s.Close() }) t.Run("existing identifier", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ Title: "test", Description: "Lorem Ipsum", Identifier: "test1", NamespaceID: 1, } err := list.Create(s, usr) assert.Error(t, err) assert.True(t, IsErrListIdentifierIsNotUnique(err)) _ = s.Close() }) t.Run("change namespace", func(t *testing.T) { t.Run("own", func(t *testing.T) { usr := &user.User{ ID: 6, Username: "user6", Email: "user6@example.com", } db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ ID: 6, Title: "Test6", Description: "Lorem Ipsum", NamespaceID: 7, // from 6 } can, err := list.CanUpdate(s, usr) assert.NoError(t, err) assert.True(t, can) err = list.Update(s, usr) assert.NoError(t, err) err = s.Commit() assert.NoError(t, err) db.AssertExists(t, "lists", map[string]interface{}{ "id": list.ID, "title": list.Title, "description": list.Description, "namespace_id": list.NamespaceID, }, false) }) // FIXME: The check for whether the namespace is archived is missing in namespace.CanWrite // t.Run("archived own", func(t *testing.T) { // db.LoadAndAssertFixtures(t) // s := db.NewSession() // list := List{ // ID: 1, // Title: "Test1", // Description: "Lorem Ipsum", // NamespaceID: 16, // from 1 // } // can, err := list.CanUpdate(s, usr) // assert.NoError(t, err) // assert.False(t, can) // namespace is archived and thus not writeable // _ = s.Close() // }) t.Run("others", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ ID: 1, Title: "Test1", Description: "Lorem Ipsum", NamespaceID: 2, // from 1 } can, _ := list.CanUpdate(s, usr) assert.False(t, can) // namespace is not writeable by us _ = s.Close() }) }) }) } func TestList_Delete(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() list := List{ ID: 1, } err := list.Delete(s, &user.User{ID: 1}) assert.NoError(t, err) err = s.Commit() assert.NoError(t, err) db.AssertMissing(t, "lists", map[string]interface{}{ "id": 1, }) } func TestList_ReadAll(t *testing.T) { t.Run("all in namespace", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() // Get all lists for our namespace lists, err := GetListsByNamespaceID(s, 1, &user.User{}) assert.NoError(t, err) assert.Equal(t, len(lists), 2) _ = s.Close() }) t.Run("all lists for user", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() u := &user.User{ID: 1} list := List{} lists3, _, _, err := list.ReadAll(s, u, "", 1, 50) assert.NoError(t, err) assert.Equal(t, reflect.TypeOf(lists3).Kind(), reflect.Slice) ls := lists3.([]*List) assert.Equal(t, 16, len(ls)) assert.Equal(t, int64(3), ls[0].ID) // List 3 has a position of 1 and should be sorted first assert.Equal(t, int64(1), ls[1].ID) assert.Equal(t, int64(4), ls[2].ID) _ = s.Close() }) t.Run("lists for nonexistant user", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() usr := &user.User{ID: 999999} list := List{} _, _, _, err := list.ReadAll(s, usr, "", 1, 50) assert.Error(t, err) assert.True(t, user.IsErrUserDoesNotExist(err)) _ = s.Close() }) t.Run("search", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() u := &user.User{ID: 1} list := List{} lists3, _, _, err := list.ReadAll(s, u, "TEST10", 1, 50) assert.NoError(t, err) ls := lists3.([]*List) assert.Equal(t, 1, len(ls)) assert.Equal(t, int64(10), ls[0].ID) _ = s.Close() }) } func TestList_ReadOne(t *testing.T) { t.Run("normal", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() defer s.Close() u := &user.User{ID: 1} l := &List{ID: 1} can, _, err := l.CanRead(s, u) assert.NoError(t, err) assert.True(t, can) err = l.ReadOne(s, u) assert.NoError(t, err) assert.Equal(t, "Test1", l.Title) }) t.Run("with subscription", func(t *testing.T) { db.LoadAndAssertFixtures(t) s := db.NewSession() defer s.Close() u := &user.User{ID: 6} l := &List{ID: 12} can, _, err := l.CanRead(s, u) assert.NoError(t, err) assert.True(t, can) err = l.ReadOne(s, u) assert.NoError(t, err) assert.NotNil(t, l.Subscription) }) }