Files
speckle-server/packages/server/modules/multiregion/services/projectRegion.ts
T
Daniel Gak Anagrov 399c998fd7 feat(multiregion): apply prepared transactions to projects (#5322)
* 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>
2025-09-04 13:07:19 +02:00

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