2020-02-07 17:27:45 +01:00
|
|
|
// Vikunja is a to-do list application to facilitate your life.
|
2020-01-19 17:52:16 +01:00
|
|
|
// Copyright 2018-2020 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 migration
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-02-18 23:00:54 +01:00
|
|
|
"code.vikunja.io/api/pkg/log"
|
2020-01-19 17:52:16 +01:00
|
|
|
"code.vikunja.io/api/pkg/models"
|
2020-01-26 18:08:06 +01:00
|
|
|
"code.vikunja.io/api/pkg/user"
|
2020-01-19 17:52:16 +01:00
|
|
|
"io/ioutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
// InsertFromStructure takes a fully nested Vikunja data structure and a user and then creates everything for this user
|
|
|
|
// (Namespaces, tasks, etc. Even attachments and relations.)
|
2020-01-26 18:08:06 +01:00
|
|
|
func InsertFromStructure(str []*models.NamespaceWithLists, user *user.User) (err error) {
|
2020-01-19 17:52:16 +01:00
|
|
|
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Creating %d namespaces", len(str))
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
// Create all namespaces
|
|
|
|
for _, n := range str {
|
|
|
|
err = n.Create(user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Created namespace %d", n.ID)
|
|
|
|
log.Debugf("[creating structure] Creating %d lists", len(n.Lists))
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
// Create all lists
|
|
|
|
for _, l := range n.Lists {
|
|
|
|
// The tasks slice is going to be reset during the creation of the list so we rescue it here to be able
|
|
|
|
// to still loop over the tasks aftere the list was created.
|
|
|
|
tasks := l.Tasks
|
|
|
|
|
|
|
|
l.NamespaceID = n.ID
|
|
|
|
err = l.Create(user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Created list %d", l.ID)
|
|
|
|
log.Debugf("[creating structure] Creating %d tasks", len(tasks))
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
// Create all tasks
|
|
|
|
for _, t := range tasks {
|
|
|
|
t.ListID = l.ID
|
|
|
|
err = t.Create(user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Created task %d", t.ID)
|
|
|
|
if len(t.RelatedTasks) > 0 {
|
|
|
|
log.Debugf("[creating structure] Creating %d related task kinds", len(t.RelatedTasks))
|
|
|
|
}
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
// Create all relation for each task
|
|
|
|
for kind, tasks := range t.RelatedTasks {
|
2020-02-18 23:00:54 +01:00
|
|
|
|
|
|
|
if len(tasks) > 0 {
|
|
|
|
log.Debugf("[creating structure] Creating %d related tasks for kind %v", len(tasks), kind)
|
|
|
|
}
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
for _, rt := range tasks {
|
2020-02-18 23:00:54 +01:00
|
|
|
// First create the related tasks if they do not exist
|
2020-01-19 17:52:16 +01:00
|
|
|
if rt.ID == 0 {
|
2020-02-18 23:00:54 +01:00
|
|
|
rt.ListID = t.ListID
|
2020-01-19 17:52:16 +01:00
|
|
|
err = rt.Create(user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Created related task %d", rt.ID)
|
2020-01-19 17:52:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Then create the relation
|
|
|
|
taskRel := &models.TaskRelation{
|
2020-02-18 23:00:54 +01:00
|
|
|
TaskID: t.ID,
|
|
|
|
OtherTaskID: rt.ID,
|
2020-01-19 17:52:16 +01:00
|
|
|
RelationKind: kind,
|
|
|
|
}
|
|
|
|
err = taskRel.Create(user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2020-02-18 23:00:54 +01:00
|
|
|
|
|
|
|
log.Debugf("[creating structure] Created task relation between task %d and %d", t.ID, rt.ID)
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create all attachments for each task
|
2020-02-18 23:00:54 +01:00
|
|
|
if len(t.Attachments) > 0 {
|
|
|
|
log.Debugf("[creating structure] Creating %d attachments", len(t.Attachments))
|
|
|
|
}
|
2020-01-19 17:52:16 +01:00
|
|
|
for _, a := range t.Attachments {
|
|
|
|
// Check if we have a file to create
|
|
|
|
if len(a.File.FileContent) > 0 {
|
|
|
|
a.TaskID = t.ID
|
|
|
|
fr := ioutil.NopCloser(bytes.NewReader(a.File.FileContent))
|
|
|
|
err = a.NewAttachment(fr, a.File.Name, a.File.Size, user)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Created new attachment %d", a.ID)
|
2020-01-19 17:52:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-18 23:00:54 +01:00
|
|
|
log.Debugf("[creating structure] Done inserting new task structure")
|
|
|
|
|
2020-01-19 17:52:16 +01:00
|
|
|
return nil
|
|
|
|
}
|