mirror of
https://github.com/hcengineering/platform.git
synced 2025-05-22 23:51:46 +00:00
Some checks are pending
CI / test (push) Blocked by required conditions
CI / build (push) Waiting to run
CI / svelte-check (push) Blocked by required conditions
CI / formatting (push) Blocked by required conditions
CI / uitest (push) Waiting to run
CI / uitest-pg (push) Waiting to run
CI / uitest-qms (push) Waiting to run
CI / uitest-workspaces (push) Waiting to run
CI / docker-build (push) Blocked by required conditions
CI / dist-build (push) Blocked by required conditions
260 lines
7.8 KiB
TypeScript
260 lines
7.8 KiB
TypeScript
import calendar from '@hcengineering/calendar'
|
|
import {
|
|
type PersonId,
|
|
type WorkspaceInfoWithStatus,
|
|
type WorkspaceUuid,
|
|
isActiveMode,
|
|
systemAccountUuid
|
|
} from '@hcengineering/core'
|
|
import { getClient as getKvsClient } from '@hcengineering/kvs-client'
|
|
import { createClient, getAccountClient } from '@hcengineering/server-client'
|
|
import { generateToken } from '@hcengineering/server-token'
|
|
import setting from '@hcengineering/setting'
|
|
import type { Db } from 'mongodb'
|
|
import { getWorkspaceTransactorEndpoint } from './utils'
|
|
|
|
interface Credentials {
|
|
refresh_token?: string | null
|
|
|
|
expiry_date?: number | null
|
|
|
|
access_token?: string | null
|
|
|
|
token_type?: string | null
|
|
|
|
id_token?: string | null
|
|
|
|
scope?: string
|
|
}
|
|
|
|
interface User extends Credentials {
|
|
userId: string
|
|
workspace: string
|
|
email: string
|
|
}
|
|
|
|
// Updated token and history types
|
|
interface UserNew extends Credentials {
|
|
userId: PersonId
|
|
workspace: WorkspaceUuid
|
|
email: string
|
|
}
|
|
|
|
interface OldHistory {
|
|
email: string
|
|
userId: string
|
|
workspace: string
|
|
historyId: string
|
|
calendarId?: string
|
|
}
|
|
|
|
interface SyncHistory {
|
|
workspace: string
|
|
timestamp: number
|
|
}
|
|
|
|
interface WorkspaceInfoProvider {
|
|
getWorkspaceInfo: (workspaceUuid: WorkspaceUuid) => Promise<WorkspaceInfoWithStatus | undefined>
|
|
}
|
|
|
|
const CALENDAR_INTEGRATION = 'google-calendar'
|
|
|
|
export async function performCalendarAccountMigrations (db: Db, region: string | null, kvsUrl: string): Promise<void> {
|
|
console.log('Start calendar migrations')
|
|
const token = generateToken(systemAccountUuid, '' as WorkspaceUuid, { service: 'tool', admin: 'true' })
|
|
const accountClient = getAccountClient(token)
|
|
|
|
const allWorkpaces = await accountClient.listWorkspaces(region)
|
|
const byId = new Map(allWorkpaces.map((it) => [it.uuid, it]))
|
|
const oldNewIds = new Map(allWorkpaces.map((it) => [it.dataId ?? it.uuid, it]))
|
|
const workspaceProvider: WorkspaceInfoProvider = {
|
|
getWorkspaceInfo: async (workspaceUuid: WorkspaceUuid) => {
|
|
const ws = oldNewIds.get(workspaceUuid as any) ?? byId.get(workspaceUuid as any)
|
|
if (ws == null) {
|
|
console.error('No workspace found for token', workspaceUuid)
|
|
return undefined
|
|
}
|
|
return ws
|
|
}
|
|
}
|
|
|
|
await migrateCalendarIntegrations(db, token, workspaceProvider)
|
|
|
|
await migrateCalendarHistory(db, token, kvsUrl, workspaceProvider)
|
|
|
|
console.log('Finished Calendar migrations')
|
|
}
|
|
|
|
const workspacePersonsMap = new Map<WorkspaceUuid, Record<string, PersonId>>()
|
|
|
|
async function getPersonIdByEmail (
|
|
workspace: WorkspaceUuid,
|
|
email: string,
|
|
oldId: string
|
|
): Promise<PersonId | undefined> {
|
|
const map = workspacePersonsMap.get(workspace)
|
|
if (map != null) {
|
|
return map[email]
|
|
} else {
|
|
const transactorUrl = await getWorkspaceTransactorEndpoint(workspace)
|
|
const token = generateToken(systemAccountUuid, workspace)
|
|
const client = await createClient(transactorUrl, token)
|
|
try {
|
|
const res: Record<string, PersonId> = {}
|
|
const integrations = await client.findAll(setting.class.Integration, {
|
|
type: calendar.integrationType.Calendar
|
|
})
|
|
for (const integration of integrations) {
|
|
const val = integration.value.trim()
|
|
if (val === '') continue
|
|
res[val] = integration.createdBy ?? integration.modifiedBy
|
|
}
|
|
workspacePersonsMap.set(workspace, res)
|
|
return res[email]
|
|
} finally {
|
|
await client.close()
|
|
}
|
|
}
|
|
}
|
|
|
|
async function migrateCalendarIntegrations (
|
|
db: Db,
|
|
token: string,
|
|
workspaceProvider: WorkspaceInfoProvider
|
|
): Promise<void> {
|
|
try {
|
|
const accountClient = getAccountClient(token)
|
|
|
|
const tokens = db.collection<User>('tokens')
|
|
|
|
const allTokens = await tokens.find({}).toArray()
|
|
|
|
for (const token of allTokens) {
|
|
try {
|
|
const ws = await workspaceProvider.getWorkspaceInfo(token.workspace as any)
|
|
if (ws == null) {
|
|
continue
|
|
}
|
|
if (!isActiveMode(ws.mode)) continue
|
|
token.workspace = ws.uuid
|
|
|
|
const personId = await getPersonIdByEmail(ws.uuid, token.email, token.userId)
|
|
if (personId == null) {
|
|
console.error('No socialId found for token', token)
|
|
continue
|
|
}
|
|
// Check/create integration in account
|
|
const existing = await accountClient.getIntegration({
|
|
kind: CALENDAR_INTEGRATION,
|
|
workspaceUuid: ws.uuid,
|
|
socialId: personId
|
|
})
|
|
|
|
if (existing == null) {
|
|
await accountClient.createIntegration({
|
|
kind: CALENDAR_INTEGRATION,
|
|
workspaceUuid: ws.uuid,
|
|
socialId: personId
|
|
})
|
|
}
|
|
|
|
const existingToken = await accountClient.getIntegrationSecret({
|
|
key: token.email,
|
|
kind: CALENDAR_INTEGRATION,
|
|
socialId: personId,
|
|
workspaceUuid: ws.uuid
|
|
})
|
|
const newToken: UserNew = {
|
|
...token,
|
|
workspace: ws?.uuid,
|
|
email: token.email,
|
|
userId: personId
|
|
}
|
|
if (existingToken == null) {
|
|
await accountClient.addIntegrationSecret({
|
|
key: newToken.email,
|
|
kind: CALENDAR_INTEGRATION,
|
|
socialId: personId,
|
|
secret: JSON.stringify(newToken),
|
|
workspaceUuid: newToken.workspace
|
|
})
|
|
} else {
|
|
const updatedToken = {
|
|
...existingToken,
|
|
...newToken
|
|
}
|
|
await accountClient.updateIntegrationSecret({
|
|
key: newToken.email,
|
|
kind: CALENDAR_INTEGRATION,
|
|
socialId: personId,
|
|
secret: JSON.stringify(updatedToken),
|
|
workspaceUuid: newToken.workspace
|
|
})
|
|
}
|
|
} catch (e) {
|
|
console.error('Error migrating token', token, e)
|
|
}
|
|
}
|
|
console.log('Calendar integrations migrations done, integration count:', allTokens.length)
|
|
} catch (e) {
|
|
console.error('Error migrating tokens', e)
|
|
}
|
|
}
|
|
|
|
async function migrateCalendarHistory (
|
|
db: Db,
|
|
token: string,
|
|
kvsUrl: string,
|
|
workspaceProvider: WorkspaceInfoProvider
|
|
): Promise<void> {
|
|
try {
|
|
console.log('Start Calendar history migrations')
|
|
const history = db.collection<OldHistory>('histories')
|
|
const allHistories = await history.find({}).toArray()
|
|
const calendarHistory = db.collection<OldHistory>('calendarHistories')
|
|
const calendarHistories = await calendarHistory.find({}).toArray()
|
|
|
|
const kvsClient = getKvsClient(CALENDAR_INTEGRATION, kvsUrl, token)
|
|
|
|
for (const history of [...calendarHistories, ...allHistories]) {
|
|
try {
|
|
const ws = await workspaceProvider.getWorkspaceInfo(history.workspace as any)
|
|
if (ws == null) {
|
|
continue
|
|
}
|
|
if (!isActiveMode(ws.mode)) continue
|
|
|
|
const personId = await getPersonIdByEmail(ws.uuid, history.email, history.userId)
|
|
if (personId == null) {
|
|
console.error('No socialId found for token', token)
|
|
continue
|
|
}
|
|
|
|
const key =
|
|
history.calendarId != null
|
|
? `${CALENDAR_INTEGRATION}:eventHistory:${ws.uuid}:${personId}:${history.email}:${history.calendarId}`
|
|
: `${CALENDAR_INTEGRATION}:calendarsHistory:${ws.uuid}:${personId}:${history.email}`
|
|
|
|
await kvsClient.setValue(key, history.historyId)
|
|
} catch (e) {
|
|
console.error('Error migrating history', history, e)
|
|
}
|
|
}
|
|
|
|
const syncHistory = db.collection<SyncHistory>('syncHistories')
|
|
const syncHistories = await syncHistory.find({}).toArray()
|
|
for (const history of syncHistories) {
|
|
const ws = await workspaceProvider.getWorkspaceInfo(history.workspace as any)
|
|
if (ws == null) {
|
|
continue
|
|
}
|
|
|
|
const key = `${CALENDAR_INTEGRATION}:calendarSync:${ws.uuid}`
|
|
await kvsClient.setValue(key, history.timestamp)
|
|
}
|
|
console.log('Finished migrating gmail history, count:', allHistories.length)
|
|
} catch (e) {
|
|
console.error('Error migrating gmail history', e)
|
|
}
|
|
}
|