Files
speckle-server/packages/server/modules/core/loaders.ts
T
Kristaps Fabians Geikins bde148f286 chore(server): migrating fully to ESM (#5042)
* wip

* some extra fixes

* stuff kinda works?

* need to figure out mocks

* need to figure out mocks

* fix db listener

* gqlgen fix

* minor gqlgen watch adjustment

* lint fixes

* delete old codegen file

* converting migrations to ESM

* getModuleDIrectory

* vitest sort of works

* added back ts-vitest

* resolve gql double load

* fixing test timeout configs

* TSC lint fix

* fix automate tests

* moar debugging

* debugging

* more debugging

* codegen update

* server works

* yargs migrated

* chore(server): getting rid of global mocks for Server ESM (#5046)

* got rid of email mock

* got rid of comment mocks

* got rid of multi region mocks

* got rid of stripe mock

* admin override mock updated

* removed final mock

* fixing import.meta.resolve calls

* another import.meta.resolve fix

* added requested test

* nyc ESM fix

* removed unneeded deps + linting

* yarn lock forgot to commit

* tryna fix flakyness

* email capture util fix

* sendEmail fix

* fix TSX check

* sender transporter fix + CR comments

* merge main fix

* test fixx

* circleci fix

* gqlgen bigint fix

* error formatter fix

* more error formatting improvements

* esmloader added to Dockerfile

* more dockerfile fixes

* bg jobs fix
2025-07-14 10:26:19 +03:00

139 lines
3.8 KiB
TypeScript

import DataLoader from 'dataloader'
import { AuthContext } from '@/modules/shared/authz'
import { graphDataloadersBuilders } from '@/modules/index'
import { ModularizedDataLoadersConstraint } from '@/modules/shared/helpers/graphqlHelper'
import { Knex } from 'knex'
import { isNonNullable, Optional } from '@speckle/shared'
import { flatten, noop, isFunction } from 'lodash-es'
import { db } from '@/db/knex'
/**
* Lets not waste memory on loaders that may not actually be invoked
*/
const makeLazyDataLoader = <K, V, C = K>(
...args: ConstructorParameters<typeof DataLoader<K, V, C>>
): DataLoader<K, V, C> => {
let dataloader: Optional<DataLoader<K, V, C>> = undefined
return new Proxy({} as DataLoader<K, V, C>, {
get(_target, prop) {
if (!dataloader) {
// If invoking clearAll() - we don't really need to do anything, no loader exists
if (prop === 'clearAll') {
return noop
}
dataloader = new DataLoader<K, V, C>(...args)
}
const ret = dataloader[prop as keyof DataLoader<K, V, C>]
if (isFunction(ret)) {
return ret.bind(dataloader)
} else {
return ret
}
}
})
}
const makeSelfClearingDataloader = <K, V, C = K>(
...args: ConstructorParameters<typeof DataLoader<K, V, C>>
) => {
const [batchLoadFn, options] = args
const dataloader = makeLazyDataLoader<K, V, C>((ids) => {
dataloader.clearAll()
return batchLoadFn(ids)
}, options)
return dataloader
}
const buildDataLoaderCreator = (selfClearing = false) => {
return <K, V, C = K>(...args: ConstructorParameters<typeof DataLoader<K, V, C>>) => {
const [batchLoadFn, options] = args
if (selfClearing) {
return makeSelfClearingDataloader<K, V, C>(batchLoadFn, {
...(options || {}),
cacheMap: null,
cache: false
})
} else {
return makeLazyDataLoader(batchLoadFn, options)
}
}
}
/**
* Build request-scoped dataloaders
* @param ctx GraphQL context w/o loaders
*/
export async function buildRequestLoaders(
ctx: AuthContext,
options?: Partial<{ cleanLoadersEarly: boolean }>
) {
const createLoader = buildDataLoaderCreator(options?.cleanLoadersEarly || false)
const modulesLoaders = await graphDataloadersBuilders()
const mainDb = db
/**
* Dataloaders autoloaded from various speckle modules, created for the specified region DB
*/
const createLoadersForRegion = (deps: { db: Knex }) => {
return {
...(Object.assign(
{},
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
...modulesLoaders.map((l) => l({ ctx, createLoader, deps }))
) as Record<string, unknown>)
} as ModularizedDataLoaders
}
const mainDbLoaders = createLoadersForRegion({ db: mainDb })
const regionLoaders = new Map<Knex, ModularizedDataLoaders>()
// Extra utilities to add on top:
/**
* Get dataloaders for specific region
*/
const forRegion = (deps: { db: Knex }) => {
if (deps.db === mainDb) {
return mainDbLoaders
}
if (!regionLoaders.has(deps.db)) {
regionLoaders.set(deps.db, createLoadersForRegion(deps))
}
return regionLoaders.get(deps.db) as ModularizedDataLoaders
}
/**
* Clear all request loaders across all regions
*/
const clearAll = () => {
const allLoaderGroups = flatten(
[mainDbLoaders, ...regionLoaders.values()].map((l) =>
Object.values(l || {}).filter(isNonNullable)
)
)
for (const groupedLoaders of allLoaderGroups) {
for (const loaderItem of Object.values(groupedLoaders)) {
loaderItem.clearAll()
}
}
}
return {
...mainDbLoaders,
clearAll,
forRegion
}
}
export interface ModularizedDataLoaders extends ModularizedDataLoadersConstraint {}
export type RequestDataLoaders = Awaited<ReturnType<typeof buildRequestLoaders>>