platform/dev/tool/src/calendar.ts
Denis Bykhov 3fcc9035bb
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
Fix calendar service (#8981)
2025-05-19 02:41:31 +07:00

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)
}
}