Files
vikunja/pkg/models/kanban_task_bucket.go

248 lines
6.7 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 models
import (
"time"
"code.vikunja.io/api/pkg/db"
"code.vikunja.io/api/pkg/events"
"code.vikunja.io/api/pkg/user"
"code.vikunja.io/api/pkg/web"
"xorm.io/xorm"
)
// TaskBucket represents the relation between a task and a kanban bucket.
// A task can only appear once per project view which is ensured by a
// unique index on the combination of task_id and project_view_id.
type TaskBucket struct {
BucketID int64 `xorm:"bigint not null index" json:"bucket_id" param:"bucket"`
Bucket *Bucket `xorm:"-" json:"bucket"`
// The task which belongs to the bucket. Together with ProjectViewID
// this field is part of a unique index to prevent duplicates.
TaskID int64 `xorm:"bigint not null index unique(task_view)" json:"task_id"`
// The view this bucket belongs to. Combined with TaskID this forms a
// unique index.
ProjectViewID int64 `xorm:"bigint not null index unique(task_view)" json:"project_view_id" param:"view"`
ProjectID int64 `xorm:"-" json:"-" param:"project"`
Task *Task `xorm:"-" json:"task"`
web.Permissions `xorm:"-" json:"-"`
web.CRUDable `xorm:"-" json:"-"`
}
func (b *TaskBucket) TableName() string {
return "task_buckets"
}
func (b *TaskBucket) CanUpdate(s *xorm.Session, a web.Auth) (bool, error) {
bucket := Bucket{
ID: b.BucketID,
ProjectID: b.ProjectID,
ProjectViewID: b.ProjectViewID,
}
return bucket.canDoBucket(s, a)
}
func (b *TaskBucket) upsert(s *xorm.Session) (err error) {
count, err := s.Where("task_id = ? AND project_view_id = ?", b.TaskID, b.ProjectViewID).
Cols("bucket_id").
Update(b)
if err != nil {
return
}
if count == 0 {
_, err = s.Insert(b)
if err != nil {
// Check if this is a unique constraint violation for the task_buckets table
if db.IsUniqueConstraintError(err, "UQE_task_buckets_task_project_view") {
return ErrTaskAlreadyExistsInBucket{
TaskID: b.TaskID,
ProjectViewID: b.ProjectViewID,
}
}
return
}
}
return
}
// updateTaskBucket is internally used to actually do the update.
func updateTaskBucket(s *xorm.Session, a web.Auth, b *TaskBucket) (err error) {
oldTaskBucket := &TaskBucket{}
_, err = s.
Where("task_id = ? AND project_view_id = ?", b.TaskID, b.ProjectViewID).
Get(oldTaskBucket)
if err != nil {
return
}
if oldTaskBucket.BucketID == b.BucketID {
// no need to do anything
return
}
view, err := GetProjectViewByIDAndProject(s, b.ProjectViewID, b.ProjectID)
if err != nil {
return err
}
bucket, err := getBucketByID(s, b.BucketID)
if err != nil {
return err
}
// If there is a bucket set, make sure they belong to the same project as the task
if view.ID != bucket.ProjectViewID {
return ErrBucketDoesNotBelongToProjectView{
ProjectViewID: view.ID,
BucketID: bucket.ID,
}
}
task := &Task{ID: b.TaskID}
err = task.ReadOne(s, a)
if err != nil {
return err
}
// Check the bucket limit
// Only check the bucket limit if the task is being moved between buckets, allow reordering the task within a bucket
if b.BucketID != 0 && b.BucketID != oldTaskBucket.BucketID {
taskCount, err := checkBucketLimit(s, a, task, bucket)
if err != nil {
return err
}
bucket.Count = taskCount
}
var updateBucket = true
// mark task done if moved into or out of the done bucket
// Only change the done state if the task's done value actually changes
var doneChanged bool
if view.DoneBucketID != 0 {
if view.DoneBucketID == b.BucketID && !task.Done {
doneChanged = true
task.Done = true
if task.isRepeating() {
oldTask := *task
oldTask.Done = false
updateDone(&oldTask, task)
updateBucket = false
b.BucketID = oldTaskBucket.BucketID
}
}
if oldTaskBucket.BucketID == view.DoneBucketID && task.Done && b.BucketID != view.DoneBucketID {
doneChanged = true
task.Done = false
}
}
if doneChanged {
if task.Done {
task.DoneAt = time.Now()
} else {
task.DoneAt = time.Time{}
}
_, err = s.Where("id = ?", task.ID).
Cols(
"done",
"due_date",
"start_date",
"end_date",
"done_at",
).
Update(task)
if err != nil {
return
}
err = task.updateReminders(s, task)
if err != nil {
return
}
// Since the done state of the task was changed, we need to move the task into all done buckets everywhere
if task.Done {
viewsWithDoneBucket := []*ProjectView{}
err = s.
Where("project_id = ? AND view_kind = ? AND bucket_configuration_mode = ? AND id != ? AND done_bucket_id != 0",
view.ProjectID, ProjectViewKindKanban, BucketConfigurationModeManual, view.ID).
Find(&viewsWithDoneBucket)
if err != nil {
return
}
for _, v := range viewsWithDoneBucket {
newBucket := &TaskBucket{
TaskID: task.ID,
ProjectViewID: v.ID,
BucketID: v.DoneBucketID,
}
err = newBucket.upsert(s)
if err != nil {
return
}
}
}
}
if updateBucket {
err = b.upsert(s)
if err != nil {
return
}
bucket.Count++
}
b.Task = task
b.Bucket = bucket
return
}
// Update is the handler to update a task bucket
// @Summary Update a task bucket
// @Description Updates a task in a bucket
// @tags task
// @Accept json
// @Produce json
// @Security JWTKeyAuth
// @Param project path int true "Project ID"
// @Param view path int true "Project View ID"
// @Param bucket path int true "Bucket ID"
// @Param taskBucket body models.TaskBucket true "The id of the task you want to move into the bucket."
// @Success 200 {object} models.TaskBucket "The updated task bucket."
// @Failure 400 {object} web.HTTPError "Invalid task bucket object provided."
// @Failure 500 {object} models.Message "Internal error"
// @Router /projects/{project}/views/{view}/buckets/{bucket}/tasks [post]
func (b *TaskBucket) Update(s *xorm.Session, a web.Auth) (err error) {
err = updateTaskBucket(s, a, b)
if err != nil {
return err
}
doer, _ := user.GetFromAuth(a)
return events.Dispatch(&TaskUpdatedEvent{
Task: b.Task,
Doer: doer,
})
}