2018-12-28 22:49:46 +01:00
// Vikunja is a todo-list application to facilitate your life.
// Copyright 2018 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 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 General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
package models
import (
"code.vikunja.io/api/pkg/log"
"code.vikunja.io/web"
"github.com/imdario/mergo"
)
// BulkTask is the definition of a bulk update task
type BulkTask struct {
2019-01-03 23:22:06 +01:00
// A list of task ids to update
2018-12-28 22:49:46 +01:00
IDs [ ] int64 ` json:"task_ids" `
Tasks [ ] * ListTask ` json:"-" `
ListTask
}
func ( bt * BulkTask ) checkIfTasksAreOnTheSameList ( ) ( err error ) {
// Get the tasks
err = bt . GetTasksByIDs ( )
if err != nil {
return err
}
if len ( bt . Tasks ) == 0 {
return ErrBulkTasksNeedAtLeastOne { }
}
// Check if all tasks are in the same list
var firstListID = bt . Tasks [ 0 ] . ListID
for _ , t := range bt . Tasks {
if t . ListID != firstListID {
return ErrBulkTasksMustBeInSameList { firstListID , t . ListID }
}
}
return nil
}
// CanUpdate checks if a user is allowed to update a task
func ( bt * BulkTask ) CanUpdate ( a web . Auth ) bool {
err := bt . checkIfTasksAreOnTheSameList ( )
if err != nil {
log . Log . Error ( "Error occurred during CanUpdate for BulkTask: %s" , err )
return false
}
doer := getUserForRights ( a )
// A user can update an task if he has write acces to its list
l := & List { ID : bt . Tasks [ 0 ] . ListID }
l . ReadOne ( )
return l . CanWrite ( doer )
}
// Update updates a bunch of tasks at once
// @Summary Update a bunch of tasks at once
// @Description Updates a bunch of tasks at once. This includes marking them as done. Note: although you could supply another ID, it will be ignored. Use task_ids instead.
// @tags task
// @Accept json
// @Produce json
2019-01-03 23:22:06 +01:00
// @Security JWTKeyAuth
2018-12-28 22:49:46 +01:00
// @Param task body models.BulkTask true "The task object. Looks like a normal task, the only difference is it uses an array of list_ids to update."
// @Success 200 {object} models.ListTask "The updated task object."
// @Failure 400 {object} code.vikunja.io/web.HTTPError "Invalid task object provided."
// @Failure 403 {object} code.vikunja.io/web.HTTPError "The user does not have access to the task (aka its list)"
// @Failure 500 {object} models.Message "Internal error"
// @Router /tasks/bulk [post]
func ( bt * BulkTask ) Update ( ) ( err error ) {
sess := x . NewSession ( )
defer sess . Close ( )
err = sess . Begin ( )
if err != nil {
return
}
for _ , oldtask := range bt . Tasks {
// When a repeating task is marked as done, we update all deadlines and reminders and set it as undone
updateDone ( oldtask , & bt . ListTask )
2018-12-29 15:29:50 +01:00
// Update the assignees
if err := oldtask . updateTaskAssignees ( bt . Assignees ) ; err != nil {
return err
}
2018-12-28 22:49:46 +01:00
// For whatever reason, xorm dont detect if done is updated, so we need to update this every time by hand
// Which is why we merge the actual task struct with the one we got from the
// The user struct overrides values in the actual one.
if err := mergo . Merge ( oldtask , & bt . ListTask , mergo . WithOverride ) ; err != nil {
return err
}
// And because a false is considered to be a null value, we need to explicitly check that case here.
2019-02-18 20:32:41 +01:00
if ! bt . ListTask . Done {
2018-12-28 22:49:46 +01:00
oldtask . Done = false
}
_ , err = sess . ID ( oldtask . ID ) .
Cols ( "text" ,
"description" ,
"done" ,
"due_date_unix" ,
"reminders_unix" ,
"repeat_after" ,
"parent_task_id" ,
"priority" ,
"start_date_unix" ,
"end_date_unix" ) .
Update ( oldtask )
if err != nil {
return sess . Rollback ( )
}
}
err = sess . Commit ( )
if err != nil {
return
}
return
}