Accomodate GQL requests caching in code generator (#237)

* Accomodate GQL requests caching in code generator

* Add GQL API request queuing
This commit is contained in:
prathamesh0 2022-11-17 00:32:08 -06:00 committed by GitHub
parent 79e903b396
commit f53371e17b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
17 changed files with 130 additions and 76 deletions

View File

@ -39,8 +39,6 @@ export const main = async (): Promise<any> => {
assert(config.server, 'Missing server config'); assert(config.server, 'Missing server config');
const { host, port } = config.server;
const { upstream, database: dbConfig, jobQueue: jobQueueConfig } = config; const { upstream, database: dbConfig, jobQueue: jobQueueConfig } = config;
assert(dbConfig, 'Missing database config'); assert(dbConfig, 'Missing database config');
@ -82,7 +80,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
return { app, server }; return { app, server };
}; };

View File

@ -3,7 +3,7 @@
// //
import assert from 'assert'; import assert from 'assert';
import { GraphQLSchema, parse, printSchema, print } from 'graphql'; import { GraphQLSchema, parse, printSchema, print, GraphQLDirective, GraphQLInt, GraphQLBoolean } from 'graphql';
import { ObjectTypeComposer, ObjectTypeComposerDefinition, ObjectTypeComposerFieldConfigMapDefinition, SchemaComposer } from 'graphql-compose'; import { ObjectTypeComposer, ObjectTypeComposerDefinition, ObjectTypeComposerFieldConfigMapDefinition, SchemaComposer } from 'graphql-compose';
import { Writable } from 'stream'; import { Writable } from 'stream';
import { utils } from 'ethers'; import { utils } from 'ethers';
@ -19,6 +19,7 @@ export class Schema {
this._composer = new SchemaComposer(); this._composer = new SchemaComposer();
this._events = []; this._events = [];
this._addGQLCacheTypes();
this._addBasicTypes(); this._addBasicTypes();
} }
@ -271,6 +272,28 @@ export class Schema {
this._composer.addSchemaMustHaveType(typeComposer); this._composer.addSchemaMustHaveType(typeComposer);
} }
_addGQLCacheTypes (): void {
// Create a enum type composer to add enum CacheControlScope in the schema composer.
const enumTypeComposer = this._composer.createEnumTC(`
enum CacheControlScope {
PUBLIC
PRIVATE
}
`);
this._composer.addSchemaMustHaveType(enumTypeComposer);
// Add the directive cacheControl in the schema composer.
this._composer.addDirective(new GraphQLDirective({
name: 'cacheControl',
locations: ['FIELD_DEFINITION', 'OBJECT', 'INTERFACE', 'UNION'],
args: {
maxAge: { type: GraphQLInt },
inheritMaxAge: { type: GraphQLBoolean },
scope: { type: enumTypeComposer.getType() }
}
}));
}
/** /**
* Adds types 'ResultEvent' and 'WatchedEvent' to the schema. * Adds types 'ResultEvent' and 'WatchedEvent' to the schema.
*/ */

View File

@ -30,6 +30,19 @@
# Use -1 for skipping check on block range. # Use -1 for skipping check on block range.
maxEventsBlockRange = 1000 maxEventsBlockRange = 1000
# GQL cache settings
[server.gqlCache]
enabled = true
# Max in-memory cache size (in bytes) (default 8 MB)
# maxCacheSize
# GQL cache-control max-age settings (in seconds)
maxAge = 15
{{#if (subgraphPath)}}
timeTravelMaxAge = 86400 # 1 day
{{/if}}
[metrics] [metrics]
host = "127.0.0.1" host = "127.0.0.1"
port = 9000 port = 9000

View File

@ -59,6 +59,18 @@
* Edit the custom hook function `createStateCheckpoint` (triggered just before default and CLI checkpoint) in [hooks.ts](./src/hooks.ts) to save the state in a `checkpoint` `State` using the `Indexer` object. * Edit the custom hook function `createStateCheckpoint` (triggered just before default and CLI checkpoint) in [hooks.ts](./src/hooks.ts) to save the state in a `checkpoint` `State` using the `Indexer` object.
### GQL Caching
To enable GQL requests caching:
* Update the `server.gqlCache` config with required settings.
* In the GQL [schema file](./src/schema.gql), use the `cacheControl` directive to apply cache hints at schema level.
* Eg. Set `inheritMaxAge` to true for non-scalar fields of a type.
* In the GQL [resolvers file](./src/resolvers.ts), uncomment the `setGQLCacheHints()` calls in resolvers for required queries.
## Run ## Run
* Run the watcher: * Run the watcher:

View File

@ -8,7 +8,7 @@ import debug from 'debug';
import Decimal from 'decimal.js'; import Decimal from 'decimal.js';
import { GraphQLResolveInfo, GraphQLScalarType } from 'graphql'; import { GraphQLResolveInfo, GraphQLScalarType } from 'graphql';
import { ValueResult, BlockHeight, gqlTotalQueryCount, gqlQueryCount, jsonBigIntStringReplacer, getResultState } from '@cerc-io/util'; import { ValueResult, BlockHeight, gqlTotalQueryCount, gqlQueryCount, jsonBigIntStringReplacer, getResultState, setGQLCacheHints } from '@cerc-io/util';
import { Indexer } from './indexer'; import { Indexer } from './indexer';
import { EventWatcher } from './events'; import { EventWatcher } from './events';
@ -22,6 +22,8 @@ const log = debug('vulcanize:resolver');
export const createResolvers = async (indexer: Indexer, eventWatcher: EventWatcher): Promise<any> => { export const createResolvers = async (indexer: Indexer, eventWatcher: EventWatcher): Promise<any> => {
assert(indexer); assert(indexer);
const gqlCacheConfig = indexer.serverConfig.gqlCache;
return { return {
BigInt: new BigInt('bigInt'), BigInt: new BigInt('bigInt'),
@ -63,14 +65,22 @@ export const createResolvers = async (indexer: Indexer, eventWatcher: EventWatch
Query: { Query: {
{{#each queries}} {{#each queries}}
{{this.name}}: (_: any, { blockHash, contractAddress {{this.name}}: (
{{~#each this.params}}, {{this.name~}} {{/each}} }: { blockHash: string, contractAddress: string _: any,
{{~#each this.params}}, {{this.name}}: {{this.type~}} {{/each}} }): Promise<ValueResult> => { { blockHash, contractAddress
{{~#each this.params}}, {{this.name~}} {{/each}} }: { blockHash: string, contractAddress: string
{{~#each this.params}}, {{this.name}}: {{this.type~}} {{/each}} },
__: any,
info: GraphQLResolveInfo
): Promise<ValueResult> => {
log('{{this.name}}', blockHash, contractAddress log('{{this.name}}', blockHash, contractAddress
{{~#each this.params}}, {{this.name~}} {{/each}}); {{~#each this.params}}, {{this.name~}} {{/each}});
gqlTotalQueryCount.inc(1); gqlTotalQueryCount.inc(1);
gqlQueryCount.labels('{{this.name}}').inc(1); gqlQueryCount.labels('{{this.name}}').inc(1);
// Set cache-control hints
// setGQLCacheHints(info, {}, gqlCacheConfig);
return indexer.{{this.name}}(blockHash, contractAddress return indexer.{{this.name}}(blockHash, contractAddress
{{~#each this.params}}, {{this.name~}} {{/each}}); {{~#each this.params}}, {{this.name~}} {{/each}});
}, },
@ -79,7 +89,7 @@ export const createResolvers = async (indexer: Indexer, eventWatcher: EventWatch
{{~#each subgraphQueries}} {{~#each subgraphQueries}}
{{this.queryName}}: async ( {{this.queryName}}: async (
_: any, _: any,
{ id, block = {} }: { id: string, block: BlockHeight }, { id, block = {} }: { id: string, block: BlockHeight },
__: any, __: any,
info: GraphQLResolveInfo info: GraphQLResolveInfo
@ -89,6 +99,9 @@ export const createResolvers = async (indexer: Indexer, eventWatcher: EventWatch
gqlQueryCount.labels('{{this.queryName}}').inc(1); gqlQueryCount.labels('{{this.queryName}}').inc(1);
assert(info.fieldNodes[0].selectionSet); assert(info.fieldNodes[0].selectionSet);
// Set cache-control hints
// setGQLCacheHints(info, block, gqlCacheConfig);
return indexer.getSubgraphEntity({{this.entityName}}, id, block, info.fieldNodes[0].selectionSet.selections); return indexer.getSubgraphEntity({{this.entityName}}, id, block, info.fieldNodes[0].selectionSet.selections);
}, },

View File

@ -39,7 +39,7 @@ export const main = async (): Promise<any> => {
const config: Config = await getConfig(argv.f); const config: Config = await getConfig(argv.f);
const { ethClient, ethProvider } = await initClients(config); const { ethClient, ethProvider } = await initClients(config);
const { host, port, kind: watcherKind } = config.server; const { kind: watcherKind } = config.server;
const db = new Database(config.database); const db = new Database(config.database);
await db.init(); await db.init();
@ -85,7 +85,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
startGQLMetricsServer(config); startGQLMetricsServer(config);

View File

@ -13,7 +13,7 @@ import { hideBin } from 'yargs/helpers';
import debug from 'debug'; import debug from 'debug';
import 'graphql-import-node'; import 'graphql-import-node';
import { DEFAULT_CONFIG_PATH, getConfig, Config, JobQueue, KIND_ACTIVE, initClients, startGQLMetricsServer, createAndStartServerWithCache } from '@cerc-io/util'; import { DEFAULT_CONFIG_PATH, getConfig, Config, JobQueue, KIND_ACTIVE, initClients, startGQLMetricsServer, createAndStartServer } from '@cerc-io/util';
import { GraphWatcher, Database as GraphDatabase } from '@cerc-io/graph-node'; import { GraphWatcher, Database as GraphDatabase } from '@cerc-io/graph-node';
import { createResolvers } from './resolvers'; import { createResolvers } from './resolvers';
@ -79,7 +79,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServerWithCache(app, typeDefs, resolvers, config.server); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
startGQLMetricsServer(config); startGQLMetricsServer(config);

View File

@ -37,7 +37,7 @@ export const main = async (): Promise<any> => {
const config: Config = await getConfig(argv.f); const config: Config = await getConfig(argv.f);
const { ethClient, ethProvider } = await initClients(config); const { ethClient, ethProvider } = await initClients(config);
const { host, port, kind: watcherKind } = config.server; const { kind: watcherKind } = config.server;
const db = new Database(config.database); const db = new Database(config.database);
await db.init(); await db.init();
@ -70,7 +70,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
return { app, server }; return { app, server };
}; };

View File

@ -36,7 +36,7 @@ export const main = async (): Promise<any> => {
const config: Config = await getConfig(argv.f); const config: Config = await getConfig(argv.f);
const { ethClient, ethProvider } = await initClients(config); const { ethClient, ethProvider } = await initClients(config);
const { host, port, kind: watcherKind } = config.server; const { kind: watcherKind } = config.server;
const db = new Database(config.database); const db = new Database(config.database);
await db.init(); await db.init();
@ -70,7 +70,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
return { app, server }; return { app, server };
}; };

View File

@ -37,7 +37,7 @@ export const main = async (): Promise<any> => {
const config: Config = await getConfig(argv.f); const config: Config = await getConfig(argv.f);
const { ethClient, ethProvider } = await initClients(config); const { ethClient, ethProvider } = await initClients(config);
const { host, port, kind: watcherKind } = config.server; const { kind: watcherKind } = config.server;
const db = new Database(config.database); const db = new Database(config.database);
await db.init(); await db.init();
@ -79,7 +79,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
return { app, server }; return { app, server };
}; };

View File

@ -36,7 +36,7 @@ export const main = async (): Promise<any> => {
const config: Config = await getConfig(argv.f); const config: Config = await getConfig(argv.f);
const { ethClient, ethProvider } = await initClients(config); const { ethClient, ethProvider } = await initClients(config);
const { host, port, kind: watcherKind } = config.server; const { kind: watcherKind } = config.server;
const db = new Database(config.database); const db = new Database(config.database);
await db.init(); await db.init();
@ -70,7 +70,7 @@ export const main = async (): Promise<any> => {
// Create an Express app // Create an Express app
const app: Application = express(); const app: Application = express();
const server = createAndStartServer(app, typeDefs, resolvers, { host, port }); const server = createAndStartServer(app, typeDefs, resolvers, config.server);
startGQLMetricsServer(config); startGQLMetricsServer(config);

View File

@ -15,6 +15,7 @@
"decimal.js": "^10.3.1", "decimal.js": "^10.3.1",
"ethers": "^5.4.4", "ethers": "^5.4.4",
"express": "^4.18.2", "express": "^4.18.2",
"express-queue": "^0.0.13",
"fs-extra": "^10.0.0", "fs-extra": "^10.0.0",
"graphql": "^15.5.0", "graphql": "^15.5.0",
"graphql-ws": "^5.11.2", "graphql-ws": "^5.11.2",

View File

@ -296,9 +296,14 @@ export const setGQLCacheHints = (info: GraphQLResolveInfo, block: BlockHeight, g
return; return;
} }
assert(gqlCache.maxAge, 'Missing server gqlCache.maxAge'); let maxAge: number;
assert(gqlCache.timeTravelMaxAge, 'Missing server gqlCache.timeTravelMaxAge'); if (_.isEmpty(block)) {
assert(gqlCache.maxAge, 'Missing server gqlCache.maxAge');
maxAge = gqlCache.maxAge;
} else {
assert(gqlCache.timeTravelMaxAge, 'Missing server gqlCache.timeTravelMaxAge');
maxAge = gqlCache.timeTravelMaxAge;
}
const maxAge = _.isEmpty(block) ? gqlCache.maxAge : gqlCache.timeTravelMaxAge;
info.cacheControl.setCacheHint({ maxAge }); info.cacheControl.setCacheHint({ maxAge });
}; };

View File

@ -7,6 +7,7 @@ import { ApolloServerPluginDrainHttpServer } from 'apollo-server-core';
import debug from 'debug'; import debug from 'debug';
import responseCachePlugin from 'apollo-server-plugin-response-cache'; import responseCachePlugin from 'apollo-server-plugin-response-cache';
import { InMemoryLRUCache } from '@apollo/utils.keyvaluecache'; import { InMemoryLRUCache } from '@apollo/utils.keyvaluecache';
import queue from 'express-queue';
import { TypeSource } from '@graphql-tools/utils'; import { TypeSource } from '@graphql-tools/utils';
import { makeExecutableSchema } from '@graphql-tools/schema'; import { makeExecutableSchema } from '@graphql-tools/schema';
@ -16,15 +17,15 @@ import { ServerConfig } from './config';
const log = debug('vulcanize:server'); const log = debug('vulcanize:server');
export const createAndStartServerWithCache = async ( export const createAndStartServer = async (
app: Application, app: Application,
typeDefs: TypeSource, typeDefs: TypeSource,
resolvers: any, resolvers: any,
serverConfig: ServerConfig serverConfig: ServerConfig
): Promise<ApolloServer> => { ): Promise<ApolloServer> => {
const host = serverConfig.host; const { host, port, gqlCache: gqlCacheConfig, maxSimultaneousRequests, maxRequestQueueLimit } = serverConfig;
const port = serverConfig.port;
const gqlCacheConfig = serverConfig.gqlCache; app.use(queue({ activeLimit: maxSimultaneousRequests || 1, queuedLimit: maxRequestQueueLimit || -1 }));
// Create HTTP server // Create HTTP server
const httpServer = createServer(app); const httpServer = createServer(app);
@ -76,50 +77,3 @@ export const createAndStartServerWithCache = async (
return server; return server;
}; };
export const createAndStartServer = async (
app: Application,
typeDefs: TypeSource,
resolvers: any,
endPoint: { host: string, port: number }
): Promise<ApolloServer> => {
// Create HTTP server
const httpServer = createServer(app);
// Create the schema
const schema = makeExecutableSchema({ typeDefs, resolvers });
// Create our WebSocket server using the HTTP server we just set up.
const wsServer = new WebSocketServer({
server: httpServer,
path: '/graphql'
});
const serverCleanup = useServer({ schema }, wsServer);
const server = new ApolloServer({
schema,
csrfPrevention: true,
plugins: [
// Proper shutdown for the HTTP server
ApolloServerPluginDrainHttpServer({ httpServer }),
// Proper shutdown for the WebSocket server
{
async serverWillStart () {
return {
async drainServer () {
await serverCleanup.dispose();
}
};
}
}
]
});
await server.start();
server.applyMiddleware({ app });
httpServer.listen(endPoint.port, endPoint.host, () => {
log(`Server is listening on ${endPoint.host}:${endPoint.port}${server.graphqlPath}`);
});
return server;
};

View File

@ -0,0 +1,6 @@
//
// Copyright 2022 Vulcanize, Inc.
//
// https://medium.com/@steveruiz/using-a-javascript-library-without-type-declarations-in-a-typescript-project-3643490015f3
declare module 'express-queue';

View File

@ -0,0 +1,6 @@
{
"name": "common",
"version": "0.1.0",
"license": "AGPL-3.0",
"typings": "main.d.ts"
}

View File

@ -5660,7 +5660,7 @@ debug@4, debug@4.3.1, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1:
dependencies: dependencies:
ms "2.1.2" ms "2.1.2"
debug@4.3.4, debug@^4.3.3: debug@4.3.4, debug@^4.3.3, debug@^4.3.4:
version "4.3.4" version "4.3.4"
resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865"
integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==
@ -7073,6 +7073,22 @@ expand-brackets@^2.1.4:
snapdragon "^0.8.1" snapdragon "^0.8.1"
to-regex "^3.0.1" to-regex "^3.0.1"
express-end@0.0.8:
version "0.0.8"
resolved "https://registry.yarnpkg.com/express-end/-/express-end-0.0.8.tgz#0c8fd95428932158f2b4cf91f4045346bf2c5323"
integrity sha512-PPntzICAq006LBpXKBVJtmRUiCRqTMZ+OB8L2RFXgx+OmkMWU66IL4DTEPF/DOcxmsuC7Y0NdbT2R71lb+pBpg==
dependencies:
debug "^2.2.0"
express-queue@^0.0.13:
version "0.0.13"
resolved "https://registry.yarnpkg.com/express-queue/-/express-queue-0.0.13.tgz#e9800d67749d4dfab7c34223f00595af933ce5df"
integrity sha512-C4OEDasGDqpXLrZICSUxbY47p5c0bKqf/3/3hwauSCmI+jVVxKBWU2w39BuKLP6nF65z87uDFBbJMPAn2ZrG3g==
dependencies:
debug "^4.3.4"
express-end "0.0.8"
mini-queue "0.0.14"
express@^4.14.0: express@^4.14.0:
version "4.17.1" version "4.17.1"
resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134" resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134"
@ -10371,6 +10387,13 @@ min-indent@^1.0.0:
resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869" resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869"
integrity sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg== integrity sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==
mini-queue@0.0.14:
version "0.0.14"
resolved "https://registry.yarnpkg.com/mini-queue/-/mini-queue-0.0.14.tgz#83d2f3f908e3cac5390bd986d1e6fbbfa0d95b93"
integrity sha512-DNh9Wn8U1jrmn1yVfpviwClyER/Y4ltgGbG+LF/KIdKJ8BEo2Q9jDDPG7tEhz6F/DTZ/ohv5D7AAXFVSFyP05Q==
dependencies:
debug "^3.1.0"
minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1: minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1:
version "1.0.1" version "1.0.1"
resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7" resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7"