2021-12-07 18:45:11 +00:00
|
|
|
//
|
|
|
|
// Copyright © 2020, 2021 Anticrm Platform Contributors.
|
|
|
|
//
|
|
|
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License. You may
|
|
|
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
//
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
|
2021-12-21 09:09:46 +00:00
|
|
|
import { AttachedDoc, Class, Client, Doc, Domain, DOMAIN_TX, Ref, Space, TxCUD, TxOperations } from '@anticrm/core'
|
2021-12-16 09:06:56 +00:00
|
|
|
import {
|
|
|
|
MigrateOperation,
|
|
|
|
MigrateUpdate,
|
|
|
|
MigrationClient,
|
|
|
|
MigrationResult,
|
|
|
|
MigrationUpgradeClient
|
|
|
|
} from '@anticrm/model'
|
2021-12-07 18:45:11 +00:00
|
|
|
import core from '@anticrm/model-core'
|
2021-12-21 09:42:18 +00:00
|
|
|
import { createDefaultKanbanTemplate, createProjectKanban, KanbanTemplate, DocWithRank, genRanks } from '@anticrm/task'
|
2021-12-15 09:04:43 +00:00
|
|
|
import { DOMAIN_TASK, DOMAIN_STATE, DOMAIN_KANBAN } from '.'
|
2021-12-07 18:45:11 +00:00
|
|
|
import task from './plugin'
|
|
|
|
|
2021-12-15 09:04:43 +00:00
|
|
|
function logInfo (msg: string, result: MigrationResult): void {
|
|
|
|
if (result.updated > 0) {
|
2021-12-16 09:06:56 +00:00
|
|
|
console.log(`Task: Migrate ${msg} ${result.updated}`)
|
2021-12-15 09:04:43 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-16 09:06:56 +00:00
|
|
|
async function migrateClass<T extends Doc> (
|
|
|
|
client: MigrationClient,
|
|
|
|
domain: Domain,
|
|
|
|
from: Ref<Class<Doc>>,
|
|
|
|
to: Ref<Class<T>>,
|
|
|
|
extraOps: MigrateUpdate<T> = {},
|
|
|
|
txExtraOps: MigrateUpdate<TxCUD<Doc>> = {}
|
|
|
|
): Promise<void> {
|
|
|
|
logInfo(`${from} => ${to}: `, await client.update<Doc>(domain, { _class: from }, { ...extraOps, _class: to }))
|
|
|
|
logInfo(
|
|
|
|
`${from} => ${to} Transactions`,
|
|
|
|
await client.update<TxCUD<Doc>>(DOMAIN_TX, { objectClass: from }, { ...txExtraOps, objectClass: to })
|
|
|
|
)
|
2021-12-15 09:04:43 +00:00
|
|
|
}
|
|
|
|
|
2021-12-07 18:45:11 +00:00
|
|
|
export const taskOperation: MigrateOperation = {
|
|
|
|
async migrate (client: MigrationClient): Promise<void> {
|
2021-12-15 09:04:43 +00:00
|
|
|
// Since we should not have Task class instances, we convert them all to Issue.
|
|
|
|
await migrateClass(client, DOMAIN_TASK, task.class.Task, task.class.Issue)
|
|
|
|
await migrateClass(client, DOMAIN_STATE, 'core:class:State' as Ref<Class<Doc>>, task.class.State)
|
2021-12-15 09:36:17 +00:00
|
|
|
await migrateClass(client, DOMAIN_STATE, 'core:class:WonState' as Ref<Class<Doc>>, task.class.WonState)
|
|
|
|
await migrateClass(client, DOMAIN_STATE, 'core:class:LostState' as Ref<Class<Doc>>, task.class.LostState)
|
2021-12-15 09:04:43 +00:00
|
|
|
await migrateClass(client, DOMAIN_KANBAN, 'view:class:Kanban' as Ref<Class<Doc>>, task.class.Kanban)
|
2021-12-16 09:06:56 +00:00
|
|
|
await migrateClass(
|
|
|
|
client,
|
|
|
|
DOMAIN_KANBAN,
|
|
|
|
'view:class:Sequence' as Ref<Class<Doc>>,
|
|
|
|
task.class.Sequence,
|
|
|
|
{ space: task.space.Sequence },
|
|
|
|
{ objectSpace: task.space.Sequence }
|
|
|
|
)
|
2021-12-15 09:04:43 +00:00
|
|
|
|
|
|
|
// Update attached to for task
|
2021-12-16 09:06:56 +00:00
|
|
|
await client.update(
|
|
|
|
DOMAIN_KANBAN,
|
|
|
|
{ _class: task.class.Sequence, attachedTo: task.class.Task },
|
|
|
|
{ attachedTo: task.class.Issue }
|
|
|
|
)
|
2021-12-15 09:04:43 +00:00
|
|
|
|
|
|
|
await migrateClass(client, DOMAIN_KANBAN, 'view:class:KanbanTemplate' as Ref<Class<Doc>>, task.class.KanbanTemplate)
|
|
|
|
await migrateClass(client, DOMAIN_KANBAN, 'view:class:StateTemplate' as Ref<Class<Doc>>, task.class.StateTemplate)
|
2021-12-16 09:06:56 +00:00
|
|
|
await migrateClass(
|
|
|
|
client,
|
|
|
|
DOMAIN_KANBAN,
|
|
|
|
'view:class:DoneStateTemplate' as Ref<Class<Doc>>,
|
|
|
|
task.class.DoneStateTemplate
|
|
|
|
)
|
2021-12-17 10:00:18 +00:00
|
|
|
await migrateClass(
|
|
|
|
client,
|
|
|
|
DOMAIN_KANBAN,
|
|
|
|
'view:class:WonStateTemplate' as Ref<Class<Doc>>,
|
|
|
|
task.class.WonStateTemplate
|
|
|
|
)
|
2021-12-16 09:06:56 +00:00
|
|
|
await migrateClass(
|
|
|
|
client,
|
|
|
|
DOMAIN_KANBAN,
|
|
|
|
'view:class:LostStateTemplate' as Ref<Class<Doc>>,
|
|
|
|
task.class.LostStateTemplate
|
|
|
|
)
|
|
|
|
|
|
|
|
await client.move(
|
|
|
|
'recruit' as Domain,
|
|
|
|
{
|
|
|
|
_class: 'recruit:class:Applicant' as Ref<Class<Doc>>
|
|
|
|
},
|
|
|
|
DOMAIN_TASK
|
|
|
|
)
|
2021-12-07 18:45:11 +00:00
|
|
|
|
2021-12-16 09:06:56 +00:00
|
|
|
await client.move(
|
|
|
|
'lead' as Domain,
|
|
|
|
{
|
|
|
|
_class: 'lead:class:Lead' as Ref<Class<Doc>>
|
|
|
|
},
|
|
|
|
DOMAIN_TASK
|
|
|
|
)
|
2021-12-15 09:04:43 +00:00
|
|
|
|
2021-12-16 09:06:56 +00:00
|
|
|
// Update done states for tasks
|
|
|
|
await client.update(DOMAIN_TASK, { _class: task.class.Issue, doneState: { $exists: false } }, { doneState: null })
|
2021-12-07 18:45:11 +00:00
|
|
|
},
|
|
|
|
async upgrade (client: MigrationUpgradeClient): Promise<void> {
|
|
|
|
console.log('Task: Performing model upgrades')
|
|
|
|
|
|
|
|
const ops = new TxOperations(client, core.account.System)
|
2021-12-16 09:06:56 +00:00
|
|
|
if ((await client.findOne(task.class.Sequence, { attachedTo: task.class.Issue })) === undefined) {
|
|
|
|
console.info('Task: Create sequence for default task project.')
|
2021-12-07 18:45:11 +00:00
|
|
|
// We need to create sequence
|
2021-12-15 09:04:43 +00:00
|
|
|
await ops.createDoc(task.class.Sequence, task.space.Sequence, {
|
|
|
|
attachedTo: task.class.Issue,
|
2021-12-07 18:45:11 +00:00
|
|
|
sequence: 0
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
console.log('Task: => sequence is ok')
|
|
|
|
}
|
2021-12-16 09:06:56 +00:00
|
|
|
if ((await client.findOne(task.class.Kanban, { attachedTo: task.space.TasksPublic })) === undefined) {
|
|
|
|
console.info('Task: Create kanban for default task project.')
|
2021-12-07 18:45:11 +00:00
|
|
|
await createProjectKanban(task.space.TasksPublic, async (_class, space, data, id) => {
|
|
|
|
const doc = await ops.findOne<Doc>(_class, { _id: id })
|
|
|
|
if (doc === undefined) {
|
|
|
|
await ops.createDoc(_class, space, data, id)
|
|
|
|
} else {
|
|
|
|
await ops.updateDoc(_class, space, id, data)
|
|
|
|
}
|
|
|
|
}).catch((err) => console.error(err))
|
|
|
|
} else {
|
|
|
|
console.log('Task: => public project Kanban is ok')
|
|
|
|
}
|
2021-12-14 13:33:30 +00:00
|
|
|
|
2021-12-21 09:42:18 +00:00
|
|
|
if (await client.findOne(core.class.TxCreateDoc, { objectId: task.template.DefaultProject }) === undefined) {
|
|
|
|
await createDefaultKanbanTemplate(async (
|
|
|
|
props,
|
|
|
|
attrs
|
|
|
|
): Promise<void> => {
|
|
|
|
await ops.createDoc(props.class, props.space, attrs, props.id)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-12-15 09:04:43 +00:00
|
|
|
console.log('View: Performing model upgrades')
|
|
|
|
|
2021-12-17 09:08:37 +00:00
|
|
|
await createMissingDoneStates(client, ops)
|
|
|
|
await updateRankItems({ client, ops, _class: task.class.State, extractOrder: (kanban) => kanban.states })
|
|
|
|
await updateRankItems({ client, ops, _class: task.class.DoneState, extractOrder: (kanban) => kanban.doneStates })
|
|
|
|
await updateRankItems({ client, ops, _class: task.class.Task, extractOrder: (kanban) => kanban.order })
|
|
|
|
await updateTemplateRankItems({ client, ops, _class: task.class.StateTemplate, extractOrder: (kanban) => kanban.states })
|
|
|
|
await updateTemplateRankItems({ client, ops, _class: task.class.DoneStateTemplate, extractOrder: (kanban) => kanban.doneStates })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function createMissingDoneStates (client: Client, ops: TxOperations): Promise<void> {
|
|
|
|
const spacesWithStates = await client.findAll(task.class.SpaceWithStates, {})
|
|
|
|
const doneStates = await client.findAll(task.class.DoneState, {})
|
|
|
|
const spaceIdsWithDoneStates = new Set(doneStates.map(x => x.space))
|
|
|
|
const outdatedSpaces = spacesWithStates.filter((space) => !spaceIdsWithDoneStates.has(space._id))
|
2021-12-15 09:04:43 +00:00
|
|
|
|
2021-12-17 09:08:37 +00:00
|
|
|
const pairRanks = [...genRanks(2)]
|
|
|
|
|
|
|
|
await Promise.all(
|
|
|
|
outdatedSpaces
|
|
|
|
.map(async (space) => {
|
|
|
|
console.log(`Creating done states for space: ${space._id}`)
|
2021-12-16 09:06:56 +00:00
|
|
|
try {
|
2021-12-17 09:08:37 +00:00
|
|
|
await Promise.all([
|
|
|
|
ops.createDoc(task.class.WonState, space._id, {
|
|
|
|
title: 'Won',
|
|
|
|
rank: pairRanks[0]
|
2021-12-16 09:06:56 +00:00
|
|
|
}),
|
2021-12-17 09:08:37 +00:00
|
|
|
ops.createDoc(task.class.LostState, space._id, {
|
|
|
|
title: 'Lost',
|
|
|
|
rank: pairRanks[1]
|
2021-12-15 09:04:43 +00:00
|
|
|
})
|
2021-12-16 09:06:56 +00:00
|
|
|
])
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e)
|
2021-12-14 13:33:30 +00:00
|
|
|
}
|
2021-12-17 09:08:37 +00:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
async function updateRankItems<T extends DocWithRank> ({
|
|
|
|
client,
|
|
|
|
ops,
|
|
|
|
_class,
|
|
|
|
extractOrder
|
|
|
|
}: {
|
|
|
|
client: Client
|
|
|
|
ops: TxOperations
|
|
|
|
_class: Ref<Class<T>>
|
|
|
|
extractOrder: (kanban: any) => Ref<T>[]
|
|
|
|
}): Promise<void> {
|
|
|
|
const allItems = await client.findAll(_class, {})
|
|
|
|
const unorderedItems = allItems
|
|
|
|
.filter((item) => item.rank === undefined)
|
|
|
|
const groupedUnsortedItems = new Map<Ref<Space>, T[]>()
|
|
|
|
|
|
|
|
unorderedItems.forEach((item) => {
|
|
|
|
const existing = groupedUnsortedItems.get(item.space) ?? []
|
|
|
|
groupedUnsortedItems.set(item.space, [...existing, item])
|
|
|
|
})
|
|
|
|
|
|
|
|
for (const [space, items] of groupedUnsortedItems.entries()) {
|
|
|
|
const kanban = await client.findOne(task.class.Kanban, { attachedTo: space })
|
|
|
|
|
|
|
|
if (kanban === undefined) {
|
|
|
|
console.error(`Failed to find kanban attached to space '${space}'`)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
const order = extractOrder(kanban)
|
|
|
|
|
|
|
|
if (order === undefined) {
|
|
|
|
console.error(`Kanban doesn't contain items order: ${kanban._id}`)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
const orderedItems = order
|
|
|
|
.map((id) => items.find(x => x._id === id))
|
|
|
|
.filter((items): items is T => items !== undefined)
|
|
|
|
const ranks = genRanks(orderedItems.length)
|
|
|
|
|
|
|
|
for (const item of orderedItems) {
|
|
|
|
const rank = ranks.next().value
|
|
|
|
|
|
|
|
if (rank === undefined) {
|
|
|
|
console.error('Failed to generate rank')
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
await ops.updateDoc(item._class as Ref<Class<DocWithRank>>, item.space, item._id, { rank })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function updateTemplateRankItems<T extends DocWithRank & AttachedDoc> ({
|
|
|
|
client,
|
|
|
|
ops,
|
|
|
|
_class,
|
|
|
|
extractOrder
|
|
|
|
}: {
|
|
|
|
client: Client
|
|
|
|
ops: TxOperations
|
|
|
|
_class: Ref<Class<T>>
|
|
|
|
extractOrder: (kanban: any) => Ref<T>[]
|
|
|
|
}): Promise<void> {
|
|
|
|
const allItems = await client.findAll(_class, {})
|
|
|
|
const unorderedItems = allItems
|
|
|
|
.filter((state) => state.rank === undefined)
|
|
|
|
const groupedUnsortedItems = new Map<Ref<Doc>, T[]>()
|
|
|
|
|
|
|
|
unorderedItems.forEach((item) => {
|
|
|
|
const existing = groupedUnsortedItems.get(item.attachedTo) ?? []
|
|
|
|
groupedUnsortedItems.set(item.attachedTo, [...existing, item])
|
|
|
|
})
|
|
|
|
|
|
|
|
for (const [attachedTo, items] of groupedUnsortedItems.entries()) {
|
|
|
|
const kanban = await client.findOne(task.class.KanbanTemplate, { _id: attachedTo as Ref<KanbanTemplate> })
|
|
|
|
|
|
|
|
if (kanban === undefined) {
|
|
|
|
console.error(`Failed to find kanban '${attachedTo}'`)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
const order = extractOrder(kanban)
|
|
|
|
|
|
|
|
if (order === undefined) {
|
|
|
|
console.error(`Kanban doesn't contain items order: ${kanban._id}`)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
const orderedItems = order
|
|
|
|
.map((id) => items.find(x => x._id === id))
|
|
|
|
.filter((items): items is T => items !== undefined)
|
|
|
|
const ranks = genRanks(orderedItems.length)
|
|
|
|
|
|
|
|
for (const item of orderedItems) {
|
|
|
|
const rank = ranks.next().value
|
|
|
|
|
|
|
|
if (rank === undefined) {
|
|
|
|
console.error('Failed to generate rank')
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
await ops.updateDoc(item._class as Ref<Class<DocWithRank>>, item.space, item._id, { rank })
|
|
|
|
}
|
2021-12-07 18:45:11 +00:00
|
|
|
}
|
|
|
|
}
|