399c998fd7
* feat(multiregion): replace user replication * chore(multiregion): optimise replication * maybe it's this * postgres is fun * once more * chore(multiregion): only replicate test user creation during multiregion tests * feat: improved replicate_query logic * fix: minor * fix: starting issue * feat: included user create and delete specs to multiregion * feat: removed console logs * fix: user defaults * fix: multiregion test helper * fix: update scenarios for users * refactor(multiregion): swap replicateQuery concept to asMultiregionOperation (#5301) feat(multiregion): introduced asMultregionOperator, refactor test to user builder classes * chore: renamings * fix: remove comments * feat: remove user replication * refactor: simplified spec usages * chore: comments * chore: branches and favs * chore: more tests * chore: more tests * fix linting * fix tests * feat: dropping replication * refactor: moved project delete to service * fix: comment * feat: updateStreamFactory and updateProjectFacotry * deleteProjectFactory + replicateFactory * deleteWorkspaceFactory * fix: selector * fix: tests * fix tests, finished createStreamFactory * feat: simplify changes * fix: remove comment * fix: minor strucutres * fix: moveProjectToRegion * fix: moved branch creation outside of multiregion scope * fix: branch creation * fix: tests * fix: ci tests * fix: removed log form test * fix: on specs, no random regionKeys * review fixes * fix: mr comments * feat: removed test --------- Co-authored-by: Charles Driesler <chuck@speckle.systems>
109 lines
3.2 KiB
TypeScript
109 lines
3.2 KiB
TypeScript
import type { Stream } from '@/modules/core/domain/streams/types'
|
|
import { ProjectNotFoundError } from '@/modules/core/errors/projects'
|
|
import { StreamNotFoundError } from '@/modules/core/errors/stream'
|
|
import type {
|
|
AsyncRegionKeyStore,
|
|
CachedRegionKeyDelete,
|
|
CachedRegionKeyLookup,
|
|
RegionKeyLookupResult,
|
|
StorageRegionKeyLookup,
|
|
StorageRegionKeyUpdate,
|
|
SyncRegionKeyLookup,
|
|
SyncRegionKeyStore
|
|
} from '@/modules/multiregion/domain/operations'
|
|
import type { EventBusEmit } from '@/modules/shared/services/eventBus'
|
|
import type { Knex } from 'knex'
|
|
|
|
export type GetProjectRegionKey = (args: {
|
|
projectId: string
|
|
}) => Promise<RegionKeyLookupResult>
|
|
|
|
export const getProjectRegionKeyFactory =
|
|
({
|
|
getRegionKeyFromMemory,
|
|
writeRegionToMemory,
|
|
getRegionKeyFromCache,
|
|
writeRegionKeyToCache,
|
|
getRegionKeyFromStorage
|
|
}: {
|
|
getRegionKeyFromMemory: SyncRegionKeyLookup
|
|
writeRegionToMemory: SyncRegionKeyStore
|
|
getRegionKeyFromCache: CachedRegionKeyLookup
|
|
writeRegionKeyToCache: AsyncRegionKeyStore
|
|
getRegionKeyFromStorage: StorageRegionKeyLookup
|
|
}): GetProjectRegionKey =>
|
|
async ({ projectId }) => {
|
|
let regionKey = getRegionKeyFromMemory({ projectId })
|
|
// if undefined, cache missed
|
|
if (regionKey !== undefined) return regionKey
|
|
|
|
regionKey = await getRegionKeyFromCache({ projectId })
|
|
// if undefined, cache missed
|
|
if (regionKey !== undefined) {
|
|
writeRegionToMemory({ projectId, regionKey })
|
|
return regionKey
|
|
}
|
|
|
|
// if this returns null, means we're in the default region
|
|
regionKey = await getRegionKeyFromStorage({ projectId })
|
|
if (regionKey === undefined) throw new StreamNotFoundError()
|
|
writeRegionToMemory({ projectId, regionKey })
|
|
await writeRegionKeyToCache({ projectId, regionKey })
|
|
return regionKey
|
|
}
|
|
|
|
export type UpdateProjectRegionKey = (args: {
|
|
projectId: string
|
|
regionKey: string
|
|
}) => Promise<Stream>
|
|
|
|
export const updateProjectRegionKeyFactory =
|
|
(deps: {
|
|
upsertProjectRegionKey: StorageRegionKeyUpdate
|
|
cacheDeleteRegionKey: CachedRegionKeyDelete
|
|
emitEvent: EventBusEmit
|
|
}): UpdateProjectRegionKey =>
|
|
async ({ projectId, regionKey }) => {
|
|
const project = await deps.upsertProjectRegionKey({
|
|
projectId,
|
|
regionKey
|
|
})
|
|
|
|
if (!project) {
|
|
throw new ProjectNotFoundError()
|
|
}
|
|
|
|
// TODO: Immediately set to new region?
|
|
await deps.cacheDeleteRegionKey({ projectId })
|
|
|
|
await deps.emitEvent({
|
|
eventName: 'multiregion.project-region-updated',
|
|
payload: {
|
|
projectId,
|
|
regionKey
|
|
}
|
|
})
|
|
|
|
return project
|
|
}
|
|
|
|
export type GetRegionDb = (args: { regionKey: string }) => Promise<Knex>
|
|
export type GetProjectDb<T extends Knex | undefined = Knex> = (args: {
|
|
projectId: string
|
|
}) => T | Promise<T>
|
|
export const getProjectDbClientFactory =
|
|
<T extends Knex | undefined>({
|
|
getProjectRegionKey,
|
|
getDefaultDb,
|
|
getRegionDb
|
|
}: {
|
|
getProjectRegionKey: GetProjectRegionKey
|
|
getDefaultDb: () => T
|
|
getRegionDb: GetRegionDb
|
|
}): GetProjectDb<T> =>
|
|
async ({ projectId }) => {
|
|
const regionKey = await getProjectRegionKey({ projectId })
|
|
if (!regionKey) return getDefaultDb()
|
|
return getRegionDb({ regionKey }) as Promise<T>
|
|
}
|