@graphprotocol/client-block-tracking
Version:
`graph-client` implements automatic block tracking using `number_gte` filter of `graph-node`. This automates the process [of fetching and tracking the block number of entites](https://thegraph.com/docs/en/developer/distributed-systems/#polling-for-updated
90 lines (89 loc) • 4.69 kB
JavaScript
import { memoize2 } from '@graphql-tools/utils';
import { getNamedType, isInputObjectType, isObjectType } from 'graphql';
import { DEFAULT_CONFIG, getNewBlockNumberFromExecutionResult, transformExecutionRequest, } from './shared.js';
const DEFAULTS = {
if: true,
validateSchema: true,
...DEFAULT_CONFIG,
};
const validateSchema = memoize2(function validateSchema(schema, config) {
const metaType = schema.getType(config.metaTypeName);
if (metaType == null || !isObjectType(metaType)) {
throw new Error(`Make sure you have a type named "${config.metaTypeName}" in this source before applying Block Tracking`);
}
const blockField = metaType.getFields()[config.blockFieldName];
if (blockField == null) {
throw new Error(`Make sure you have a type named "${config.metaTypeName}" with "${config.blockFieldName}" field in this source before applying Block Tracking`);
}
const blockType = getNamedType(blockField.type);
if (!isObjectType(blockType)) {
throw new Error(`Make sure you have a correct block type in this source before applying Block Tracking`);
}
const blockNumberField = blockType.getFields()[config.blockNumberFieldName];
if (blockNumberField == null) {
throw new Error(`Make sure you have a correct block type with "${config.blockNumberFieldName}" field in this source before applying Block Tracking`);
}
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();
const metaQueryField = queryFields[config.metaRootFieldName];
if (metaQueryField == null) {
throw new Error(`Make sure you have a query type with "${config.metaRootFieldName}" field in this source before applying Block Tracking`);
}
const metaQueryFieldType = getNamedType(metaQueryField.type);
if (!isObjectType(metaQueryFieldType) || metaQueryFieldType.name !== config.metaTypeName) {
throw new Error(`Make sure you have a query type with "${config.metaRootFieldName}" field with the correct ${config.metaTypeName} type in this source before applying Block Tracking`);
}
for (const fieldName in queryFields) {
if (fieldName === config.metaRootFieldName) {
continue;
}
const field = queryFields[fieldName];
const blockArgument = field.args.find((arg) => arg.name === config.blockArgumentName);
if (blockArgument == null) {
throw new Error(`Make sure you have query root fields with "${config.blockArgumentName}" argument in this source before applying Block Tracking`);
}
const blockArgumentType = getNamedType(blockArgument.type);
if (!isInputObjectType(blockArgumentType)) {
throw new Error(`Make sure you have query root fields with "${config.blockArgumentName}" argument returning correct type in this source before applying Block Tracking`);
}
const blockArgumentFields = blockArgumentType.getFields();
const minBlockArgument = blockArgumentFields[config.minBlockArgumentName];
if (minBlockArgument == null) {
throw new Error(`Make sure you have query root fields with "${config.blockArgumentName}" argument with "${config.minBlockArgumentName}" field in this source before applying Block Tracking`);
}
}
});
const schemaMinBlockMap = new WeakMap();
export class BlockTrackingTransform {
constructor({ config } = {}) {
this.config = {
...DEFAULTS,
...config,
};
if (!this.config.if) {
return {};
}
}
transformSchema(schema, subschemaConfig) {
if (this.config.validateSchema) {
validateSchema(subschemaConfig.schema, this.config);
}
return schema;
}
transformRequest(executionRequest, delegationContext) {
return transformExecutionRequest(executionRequest, this.config, delegationContext.transformedSchema, delegationContext.subschemaConfig?.batch, schemaMinBlockMap.get(delegationContext.subschema));
}
transformResult(originalResult, delegationContext) {
const newBlockNumber = getNewBlockNumberFromExecutionResult(originalResult, this.config);
if (newBlockNumber != null) {
const existingMinBlock = schemaMinBlockMap.get(delegationContext.subschema);
if (existingMinBlock == null || newBlockNumber > existingMinBlock) {
schemaMinBlockMap.set(delegationContext.subschema, newBlockNumber);
}
}
return originalResult;
}
}