0cafe77013
* chore(regions): failing test for moving blobs * fix(blobs): use blob storage table * chore(regions): include file_uploads table for thoroughness * fix(regions): extract repeated string to utils * chore(regions): move validation as own service * feat(regions): update project region key after move * chore(regions): update tests to make gql requests
109 lines
3.1 KiB
TypeScript
109 lines
3.1 KiB
TypeScript
import { Stream } from '@/modules/core/domain/streams/types'
|
|
import { ProjectNotFoundError } from '@/modules/core/errors/projects'
|
|
import { StreamNotFoundError } from '@/modules/core/errors/stream'
|
|
import {
|
|
AsyncRegionKeyStore,
|
|
CachedRegionKeyDelete,
|
|
CachedRegionKeyLookup,
|
|
RegionKeyLookupResult,
|
|
StorageRegionKeyLookup,
|
|
StorageRegionKeyUpdate,
|
|
SyncRegionKeyLookup,
|
|
SyncRegionKeyStore
|
|
} from '@/modules/multiregion/domain/operations'
|
|
import { EventBusEmit } from '@/modules/shared/services/eventBus'
|
|
import { 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>
|
|
type GetDefaultDb = () => Knex
|
|
|
|
export type GetProjectDb = (args: { projectId: string }) => Promise<Knex>
|
|
export const getProjectDbClientFactory =
|
|
({
|
|
getProjectRegionKey,
|
|
getDefaultDb,
|
|
getRegionDb
|
|
}: {
|
|
getProjectRegionKey: GetProjectRegionKey
|
|
getDefaultDb: GetDefaultDb
|
|
getRegionDb: GetRegionDb
|
|
}): GetProjectDb =>
|
|
async ({ projectId }) => {
|
|
const regionKey = await getProjectRegionKey({ projectId })
|
|
if (!regionKey) return getDefaultDb()
|
|
return getRegionDb({ regionKey })
|
|
}
|