UNPKG

@graphprotocol/client-auto-pagination

Version:

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

237 lines (236 loc) • 12.2 kB
import { delegateToSchema } from '@graphql-tools/delegate'; import { isListType, isNonNullType, Kind, visit, } from 'graphql'; import { memoize2 } from '@graphql-tools/utils'; import _ from 'lodash'; const DEFAULTS = { if: true, validateSchema: true, limitOfRecords: 1000, firstArgumentName: 'first', skipArgumentName: 'skip', lastIdArgumentName: 'where.id_gte', skipArgumentLimit: 5000, }; 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) { const document = visit(executionRequest.document, { SelectionSet: { leave: (selectionSet) => { const newSelections = []; for (const selectionNode of selectionSet.selections) { if (selectionNode.kind === Kind.FIELD && !selectionNode.name.value.startsWith('_') && !selectionNode.arguments?.some((argNode) => argNode.name.value === 'id')) { const existingArgs = []; let firstArg; let skipArg; for (const existingArg of selectionNode.arguments ?? []) { if (existingArg.name.value === this.config.firstArgumentName) { firstArg = existingArg; } else if (existingArg.name.value === this.config.skipArgumentName) { skipArg = existingArg; } else { existingArgs.push(existingArg); } } if (firstArg != null) { let numberOfTotalRecords; if (firstArg.value.kind === Kind.INT) { numberOfTotalRecords = parseInt(firstArg.value.value); } else if (firstArg.value.kind === Kind.VARIABLE) { numberOfTotalRecords = executionRequest.variables?.[firstArg.value.name.value]; delete executionRequest.variables?.[firstArg.value.name.value]; } if (numberOfTotalRecords != null && numberOfTotalRecords > this.config.limitOfRecords) { const fieldName = selectionNode.name.value; const aliasName = selectionNode.alias?.value || fieldName; let initialSkip = 0; if (skipArg?.value?.kind === Kind.INT) { initialSkip = parseInt(skipArg.value.value); } else if (skipArg?.value?.kind === Kind.VARIABLE) { initialSkip = executionRequest.variables?.[skipArg.value.name.value]; delete executionRequest.variables?.[skipArg.value.name.value]; } let skip; for (skip = initialSkip; numberOfTotalRecords - skip + initialSkip > 0; skip += this.config.limitOfRecords) { newSelections.push({ ...selectionNode, alias: { kind: Kind.NAME, value: `splitted_${skip}_${aliasName}`, }, arguments: [ ...existingArgs, { kind: Kind.ARGUMENT, name: { kind: Kind.NAME, value: this.config.firstArgumentName, }, value: { kind: Kind.INT, value: Math.min(numberOfTotalRecords - skip + initialSkip, this.config.limitOfRecords).toString(), }, }, { kind: Kind.ARGUMENT, name: { kind: Kind.NAME, value: this.config.skipArgumentName, }, value: { kind: Kind.INT, value: skip.toString(), }, }, ], }); } continue; } } } newSelections.push(selectionNode); } return { ...selectionSet, selections: newSelections, }; }, }, }); delete delegationContext.args; return { ...executionRequest, document, }; } transformResult(originalResult) { if (originalResult.data != null) { return { ...originalResult, data: mergeSplittedResults(originalResult.data), }; } return originalResult; } } function mergeSplittedResults(originalData) { if (originalData != null && typeof originalData === 'object') { if (Array.isArray(originalData)) { return originalData.map((record) => mergeSplittedResults(record)); } const finalData = {}; for (const fullAliasName in originalData) { if (fullAliasName.startsWith('splitted_')) { const [, , ...rest] = fullAliasName.split('_'); const aliasName = rest.join('_'); finalData[aliasName] = finalData[aliasName] || []; for (const record of originalData[fullAliasName]) { finalData[aliasName].push(mergeSplittedResults(record)); } } else { finalData[fullAliasName] = mergeSplittedResults(originalData[fullAliasName]); } } return finalData; } return originalData; }