@graphql-tools/batch-delegate
Version:
A set of utils for faster development of GraphQL tools
131 lines (126 loc) • 4.42 kB
JavaScript
import { getActualFieldNodes, delegateToSchema } from '@graphql-tools/delegate';
import { memoize2, memoize1, relocatedError } from '@graphql-tools/utils';
import { fakePromise, handleMaybePromise } from '@whatwg-node/promise-helpers';
import DataLoader from 'dataloader';
import { getNamedType, print, GraphQLList } from 'graphql';
const DEFAULT_ARGS_FROM_KEYS = (keys) => ({ ids: keys });
function createBatchFn(options) {
const argsFromKeys = options.argsFromKeys ?? DEFAULT_ARGS_FROM_KEYS;
const fieldName = options.fieldName ?? options.info.fieldName;
const { valuesFromResults, lazyOptionsFn } = options;
return function batchFn(keys) {
return fakePromise(
handleMaybePromise(
() => delegateToSchema({
returnType: new GraphQLList(
getNamedType(options.returnType || options.info.returnType)
),
onLocatedError: (originalError) => {
if (originalError.path == null) {
return originalError;
}
const [pathFieldName, pathNumber, ...rest] = originalError.path;
if (pathFieldName !== fieldName) {
return originalError;
}
const pathNumberType = typeof pathNumber;
if (pathNumberType !== "number") {
return originalError;
}
return relocatedError(originalError, [fieldName, ...rest]);
},
args: argsFromKeys(keys),
...lazyOptionsFn == null ? options : lazyOptionsFn(options, keys)
}),
(results) => {
if (results instanceof Error) {
return keys.map(() => results);
}
const values = valuesFromResults == null ? results : valuesFromResults(results, keys);
return Array.isArray(values) ? values : keys.map(() => values);
}
)
);
};
}
const getLoadersMap = memoize2(function getLoadersMap2(_context, _schema) {
return /* @__PURE__ */ new Map();
});
const GLOBAL_CONTEXT = {};
const memoizedJsonStringify = memoize1(function jsonStringify(value) {
return JSON.stringify(value);
});
const memoizedPrint = memoize1(print);
function defaultCacheKeyFn(key) {
if (typeof key === "object") {
return memoizedJsonStringify(key);
}
return key;
}
function getLoader(options) {
const {
schema,
context,
info,
fieldName = info.fieldName,
dataLoaderOptions,
fieldNodes = info.fieldNodes[0] && getActualFieldNodes(info.fieldNodes[0]),
selectionSet = fieldNodes?.[0]?.selectionSet,
returnType = info.returnType,
argsFromKeys = DEFAULT_ARGS_FROM_KEYS,
key
} = options;
const loaders = getLoadersMap(context ?? GLOBAL_CONTEXT, schema);
let cacheKey = fieldName;
if (returnType) {
const namedType = getNamedType(returnType);
cacheKey += "@" + namedType.name;
}
if (selectionSet != null) {
cacheKey += memoizedPrint(selectionSet);
}
const fieldNode = fieldNodes?.[0];
if (fieldNode?.arguments) {
const args = argsFromKeys([key]);
cacheKey += fieldNode.arguments.filter((arg) => arg.name.value in args).map((arg) => memoizedPrint(arg)).join(",");
}
let loader = loaders.get(cacheKey);
if (loader === void 0) {
const batchFn = createBatchFn(options);
loader = new DataLoader(batchFn, {
// Prevents the keys to be passed with the same structure
cacheKeyFn: defaultCacheKeyFn,
...dataLoaderOptions
});
loaders.set(cacheKey, loader);
}
return loader;
}
function batchDelegateToSchema(options) {
const key = options.key;
if (key == null) {
return null;
} else if (Array.isArray(key) && !key.length) {
return [];
}
const loader = getLoader(options);
return Array.isArray(key) ? loader.loadMany(key) : loader.load(key);
}
function createBatchDelegateFn(optionsOrArgsFromKeys, lazyOptionsFn, dataLoaderOptions, valuesFromResults) {
return typeof optionsOrArgsFromKeys === "function" ? createBatchDelegateFnImpl({
argsFromKeys: optionsOrArgsFromKeys,
lazyOptionsFn,
dataLoaderOptions,
valuesFromResults
}) : createBatchDelegateFnImpl(optionsOrArgsFromKeys);
}
function createBatchDelegateFnImpl(options) {
return (batchDelegateOptions) => {
const loader = getLoader({
...options,
...batchDelegateOptions
});
return loader.load(batchDelegateOptions.key);
};
}
export { batchDelegateToSchema, createBatchDelegateFn };