2021-02-17 15:55:59 -08:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
|
|
// See LICENSE.txt for license information.
|
2021-02-17 12:03:45 -08:00
|
|
|
import * as fs from 'fs'
|
|
|
|
import minimist from 'minimist'
|
|
|
|
import {exit} from 'process'
|
2022-02-14 10:38:45 -05:00
|
|
|
import {ArchiveUtils} from '../util/archive'
|
2021-08-06 01:40:02 -06:00
|
|
|
import {Block} from '../../webapp/src/blocks/block'
|
2022-03-31 22:02:33 -04:00
|
|
|
import {Board} from '../../webapp/src/blocks/board'
|
2021-08-06 01:40:02 -06:00
|
|
|
import {IPropertyOption, IPropertyTemplate, createBoard} from '../../webapp/src/blocks/board'
|
|
|
|
import {createBoardView} from '../../webapp/src/blocks/boardView'
|
|
|
|
import {createCard} from '../../webapp/src/blocks/card'
|
|
|
|
import {createTextBlock} from '../../webapp/src/blocks/textBlock'
|
2021-02-17 12:03:45 -08:00
|
|
|
import {Asana, Workspace} from './asana'
|
|
|
|
import {Utils} from './utils'
|
|
|
|
|
|
|
|
// HACKHACK: To allow Utils.CreateGuid to work
|
|
|
|
(global.window as any) = {}
|
|
|
|
|
2021-02-22 10:54:41 -08:00
|
|
|
const optionColors = [
|
|
|
|
// 'propColorDefault',
|
|
|
|
'propColorGray',
|
|
|
|
'propColorBrown',
|
|
|
|
'propColorOrange',
|
|
|
|
'propColorYellow',
|
|
|
|
'propColorGreen',
|
|
|
|
'propColorBlue',
|
|
|
|
'propColorPurple',
|
|
|
|
'propColorPink',
|
|
|
|
'propColorRed',
|
|
|
|
]
|
|
|
|
let optionColorIndex = 0
|
|
|
|
|
2021-02-17 12:03:45 -08:00
|
|
|
function main() {
|
|
|
|
const args: minimist.ParsedArgs = minimist(process.argv.slice(2))
|
|
|
|
|
|
|
|
const inputFile = args['i']
|
|
|
|
const outputFile = args['o'] || 'archive.focalboard'
|
|
|
|
|
|
|
|
if (!inputFile) {
|
|
|
|
showHelp()
|
|
|
|
}
|
|
|
|
|
2021-02-17 15:55:59 -08:00
|
|
|
if (!fs.existsSync(inputFile)) {
|
|
|
|
console.error(`File not found: ${inputFile}`)
|
|
|
|
exit(2)
|
|
|
|
}
|
|
|
|
|
2021-02-17 12:03:45 -08:00
|
|
|
// Read input
|
|
|
|
const inputData = fs.readFileSync(inputFile, 'utf-8')
|
|
|
|
const input = JSON.parse(inputData) as Asana
|
|
|
|
|
|
|
|
// Convert
|
2022-03-31 22:02:33 -04:00
|
|
|
const [boards, blocks] = convert(input)
|
2021-02-17 12:03:45 -08:00
|
|
|
|
|
|
|
// Save output
|
2021-03-02 13:21:55 -08:00
|
|
|
// TODO: Stream output
|
2022-03-31 22:02:33 -04:00
|
|
|
const outputData = ArchiveUtils.buildBlockArchive(boards, blocks)
|
2021-02-17 12:03:45 -08:00
|
|
|
fs.writeFileSync(outputFile, outputData)
|
|
|
|
|
|
|
|
console.log(`Exported to ${outputFile}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
function getProjects(input: Asana): Workspace[] {
|
2021-02-17 15:55:59 -08:00
|
|
|
const projectMap = new Map<string, Workspace>()
|
2021-02-17 12:03:45 -08:00
|
|
|
|
2021-02-17 15:55:59 -08:00
|
|
|
input.data.forEach(datum => {
|
|
|
|
datum.projects.forEach(project => {
|
|
|
|
if (!projectMap.get(project.gid)) {
|
|
|
|
projectMap.set(project.gid, project)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2021-02-17 12:03:45 -08:00
|
|
|
|
2021-02-17 15:55:59 -08:00
|
|
|
return [...projectMap.values()]
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
function getSections(input: Asana, projectId: string): Workspace[] {
|
2021-02-17 15:55:59 -08:00
|
|
|
const sectionMap = new Map<string, Workspace>()
|
|
|
|
|
|
|
|
input.data.forEach(datum => {
|
|
|
|
const membership = datum.memberships.find(o => o.project.gid === projectId)
|
|
|
|
if (membership) {
|
|
|
|
if (!sectionMap.get(membership.section.gid)) {
|
|
|
|
sectionMap.set(membership.section.gid, membership.section)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return [...sectionMap.values()]
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
|
2022-03-31 22:02:33 -04:00
|
|
|
function convert(input: Asana): [Board[], Block[]] {
|
2021-02-17 15:55:59 -08:00
|
|
|
const projects = getProjects(input)
|
|
|
|
if (projects.length < 1) {
|
|
|
|
console.error('No projects found')
|
2022-03-31 22:02:33 -04:00
|
|
|
return [[],[]]
|
2021-02-17 15:55:59 -08:00
|
|
|
}
|
2021-02-17 12:03:45 -08:00
|
|
|
|
2021-02-17 15:55:59 -08:00
|
|
|
// TODO: Handle multiple projects
|
|
|
|
const project = projects[0]
|
2021-02-17 12:03:45 -08:00
|
|
|
|
2022-03-31 22:02:33 -04:00
|
|
|
const boards: Board[] = []
|
2021-08-06 01:40:02 -06:00
|
|
|
const blocks: Block[] = []
|
2021-02-17 12:03:45 -08:00
|
|
|
|
|
|
|
// Board
|
2021-08-06 01:40:02 -06:00
|
|
|
const board = createBoard()
|
2021-02-17 12:03:45 -08:00
|
|
|
console.log(`Board: ${project.name}`)
|
|
|
|
board.title = project.name
|
|
|
|
|
|
|
|
// Convert sections (columns) to a Select property
|
|
|
|
const optionIdMap = new Map<string, string>()
|
|
|
|
const options: IPropertyOption[] = []
|
2021-02-17 15:55:59 -08:00
|
|
|
const sections = getSections(input, project.gid)
|
2021-02-17 12:03:45 -08:00
|
|
|
sections.forEach(section => {
|
|
|
|
const optionId = Utils.createGuid()
|
|
|
|
optionIdMap.set(section.gid, optionId)
|
2021-02-22 10:54:41 -08:00
|
|
|
const color = optionColors[optionColorIndex % optionColors.length]
|
|
|
|
optionColorIndex += 1
|
2021-02-17 12:03:45 -08:00
|
|
|
const option: IPropertyOption = {
|
|
|
|
id: optionId,
|
|
|
|
value: section.name,
|
2021-02-22 10:54:41 -08:00
|
|
|
color,
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
options.push(option)
|
|
|
|
})
|
|
|
|
|
|
|
|
const cardProperty: IPropertyTemplate = {
|
|
|
|
id: Utils.createGuid(),
|
|
|
|
name: 'Section',
|
|
|
|
type: 'select',
|
|
|
|
options
|
|
|
|
}
|
2022-03-22 15:24:34 +01:00
|
|
|
board.cardProperties = [cardProperty]
|
2022-03-31 22:02:33 -04:00
|
|
|
boards.push(board)
|
2021-02-17 12:03:45 -08:00
|
|
|
|
|
|
|
// Board view
|
2021-08-06 01:40:02 -06:00
|
|
|
const view = createBoardView()
|
2021-02-17 12:03:45 -08:00
|
|
|
view.title = 'Board View'
|
2021-08-06 01:40:02 -06:00
|
|
|
view.fields.viewType = 'board'
|
2021-02-17 12:03:45 -08:00
|
|
|
view.parentId = board.id
|
2022-03-31 22:02:33 -04:00
|
|
|
view.boardId = board.id
|
2021-02-17 12:03:45 -08:00
|
|
|
blocks.push(view)
|
|
|
|
|
|
|
|
// Cards
|
|
|
|
input.data.forEach(card => {
|
|
|
|
console.log(`Card: ${card.name}`)
|
|
|
|
|
2021-08-06 01:40:02 -06:00
|
|
|
const outCard = createCard()
|
2021-02-17 12:03:45 -08:00
|
|
|
outCard.title = card.name
|
2022-03-31 22:02:33 -04:00
|
|
|
outCard.boardId = board.id
|
2021-02-17 12:03:45 -08:00
|
|
|
outCard.parentId = board.id
|
|
|
|
|
|
|
|
// Map lists to Select property options
|
2021-02-17 15:55:59 -08:00
|
|
|
const membership = card.memberships.find(o => o.project.gid === project.gid)
|
2021-02-17 12:03:45 -08:00
|
|
|
if (membership) {
|
|
|
|
const optionId = optionIdMap.get(membership.section.gid)
|
|
|
|
if (optionId) {
|
2021-08-06 01:40:02 -06:00
|
|
|
outCard.fields.properties[cardProperty.id] = optionId
|
2021-02-17 12:03:45 -08:00
|
|
|
} else {
|
|
|
|
console.warn(`Invalid idList: ${membership.section.gid} for card: ${card.name}`)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
console.warn(`Missing idList for card: ${card.name}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
blocks.push(outCard)
|
|
|
|
|
|
|
|
if (card.notes) {
|
|
|
|
// console.log(`\t${card.notes}`)
|
2021-08-06 01:40:02 -06:00
|
|
|
const text = createTextBlock()
|
2021-02-17 12:03:45 -08:00
|
|
|
text.title = card.notes
|
|
|
|
text.parentId = outCard.id
|
2022-03-31 22:02:33 -04:00
|
|
|
text.boardId = board.id
|
2021-02-17 12:03:45 -08:00
|
|
|
blocks.push(text)
|
|
|
|
|
2021-08-06 01:40:02 -06:00
|
|
|
outCard.fields.contentOrder = [text.id]
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
console.log('')
|
|
|
|
console.log(`Found ${input.data.length} card(s).`)
|
|
|
|
|
2022-03-31 22:02:33 -04:00
|
|
|
return [boards, blocks]
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
function showHelp() {
|
|
|
|
console.log('import -i <input.json> -o [output.focalboard]')
|
2021-02-17 15:55:59 -08:00
|
|
|
exit(1)
|
2021-02-17 12:03:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
main()
|