UNPKG

@graphprotocol/client-auto-pagination

Version:

`graph-client` implements automatic pagination using `first:` and `after:` filters of `graph-node`.

112 lines (111 loc) 5.53 kB
import { delegateToSchema } from '@graphql-tools/delegate'; import { isListType, isNonNullType, } from 'graphql'; import { memoize2 } from '@graphql-tools/utils'; import _ from 'lodash'; import { DEFAULT_OPTIONS, transformExecutionRequest, transformExecutionResponse } from './shared.js'; const DEFAULTS = { if: true, validateSchema: true, ...DEFAULT_OPTIONS, }; const validateSchema = memoize2(function validateSchema(schema, config) { const queryType = schema.getQueryType(); if (queryType == null) { throw new Error(`Make sure you have a query type in this source before applying Block Tracking`); } const queryFields = queryType.getFields(); for (const fieldName in queryFields) { if (fieldName.startsWith('_')) { continue; } const field = queryFields[fieldName]; const nullableType = isNonNullType(field.type) ? field.type.ofType : field.type; if (isListType(nullableType)) { if (!field.args.some((arg) => arg.name === config.firstArgumentName)) { throw new Error(`Make sure you have a ${config.firstArgumentName} argument in the query field ${fieldName}`); } if (!field.args.some((arg) => arg.name === config.skipArgumentName)) { throw new Error(`Make sure you have a ${config.skipArgumentName} argument in the query field ${fieldName}`); } } } }); /* const getQueryFieldNames = memoize1(function getQueryFields(schema: GraphQLSchema) { const queryType = schema.getQueryType() if (queryType == null) { throw new Error(`Make sure you have a query type in this source before applying Block Tracking`) } return Object.keys(queryType.getFields()) }) */ export default class AutoPaginationTransform { constructor({ config } = {}) { this.config = { ...DEFAULTS, ...config }; if (this.config.if === false) { return {}; } } transformSchema(schema, subschemaConfig) { if (this.config.validateSchema) { validateSchema(subschemaConfig.schema, this.config); } if (schema != null) { const queryType = schema.getQueryType(); if (queryType != null) { const queryFields = queryType.getFields(); for (const fieldName in queryFields) { if (!fieldName.startsWith('_')) { const field = queryFields[fieldName]; const existingResolver = field.resolve; field.resolve = async (root, args, context, info) => { const totalRecords = args[this.config.firstArgumentName] || this.config.limitOfRecords; const initialSkipValue = args[this.config.skipArgumentName] || 0; if (totalRecords >= this.config.skipArgumentLimit * 2) { let remainingRecords = totalRecords; const records = []; while (remainingRecords > 0) { let skipValue = records.length === 0 ? initialSkipValue : 0; const lastIdValue = records.length > 0 ? records[records.length - 1].id : null; while (skipValue < this.config.skipArgumentLimit && remainingRecords > 0) { const newArgs = { ...args, }; if (lastIdValue) { _.set(newArgs, this.config.lastIdArgumentName, lastIdValue); } _.set(newArgs, this.config.skipArgumentName, skipValue); const askedRecords = Math.min(remainingRecords, this.config.skipArgumentLimit); _.set(newArgs, this.config.firstArgumentName, askedRecords); const result = await delegateToSchema({ schema, fieldName, args: newArgs, context, info, }); if (!Array.isArray(result)) { return result; } records.push(...result); skipValue += askedRecords; remainingRecords -= askedRecords; } } return records; } return existingResolver(root, args, context, info); }; } } } } return schema; } transformRequest(executionRequest, delegationContext) { return transformExecutionRequest(executionRequest, this.config, delegationContext); } transformResult(originalResult) { return transformExecutionResponse(originalResult); } } export { useAutoPagination } from './plugin.js';