@graphprotocol/client-polling-live
Version:
98 lines (97 loc) • 4.77 kB
JavaScript
import { isAsyncIterable } from '@envelop/core';
import { GraphQLError, Kind, visit } from 'graphql';
import { Repeater } from '@repeaterjs/repeater';
import { mergeSchemas } from '@graphql-tools/schema';
export default function usePollingLive({ config: { defaultInterval = 1000, defaultPauseOnBackground = true } = {}, } = {}) {
return {
onSchemaChange({ schema, replaceSchema }) {
if (!schema.getDirective('live')) {
replaceSchema(mergeSchemas({
schemas: [schema],
typeDefs: /* GraphQL */ `
directive @live(interval: Int = ${defaultInterval}, pauseOnBackground: Boolean = ${defaultPauseOnBackground}) on QUERY
`,
assumeValid: true,
assumeValidSDL: true,
}));
}
},
onExecute({ args, executeFn, setExecuteFn }) {
var _a, _b, _c, _d;
let liveDirectiveNode;
args.document = visit(args.document, {
OperationDefinition(node) {
var _a;
if (args.operationName != null && ((_a = node.name) === null || _a === void 0 ? void 0 : _a.value) !== args.operationName) {
return;
}
const directives = [];
if (node.directives && node.operation === 'query') {
for (const directive of node.directives) {
if (directive.name.value === 'live') {
liveDirectiveNode = directive;
}
else {
directives.push(directive);
}
}
return {
...node,
directives,
};
}
return node;
},
});
if (liveDirectiveNode) {
const intervalArgNode = (_a = liveDirectiveNode.arguments) === null || _a === void 0 ? void 0 : _a.find((argNode) => argNode.name.value === 'interval');
let pauseOnBackground = defaultPauseOnBackground;
const pauseOnBackgroundArgNode = (_b = liveDirectiveNode.arguments) === null || _b === void 0 ? void 0 : _b.find((argNode) => argNode.name.value === 'pauseOnBackground');
if (((_c = pauseOnBackgroundArgNode === null || pauseOnBackgroundArgNode === void 0 ? void 0 : pauseOnBackgroundArgNode.value) === null || _c === void 0 ? void 0 : _c.kind) === Kind.BOOLEAN) {
pauseOnBackground = pauseOnBackgroundArgNode.value.value;
}
let intervalMs = defaultInterval;
if (((_d = intervalArgNode === null || intervalArgNode === void 0 ? void 0 : intervalArgNode.value) === null || _d === void 0 ? void 0 : _d.kind) === Kind.INT) {
intervalMs = parseInt(intervalArgNode.value.value);
}
function checkHidden() {
var _a, _b;
if (!pauseOnBackground) {
return false;
}
return (_b = (_a = globalThis.document) === null || _a === void 0 ? void 0 : _a.hidden) !== null && _b !== void 0 ? _b : false;
}
setExecuteFn((args) => new Repeater((push, stop) => {
let finished = false;
async function pump() {
if (finished) {
return;
}
if (!checkHidden()) {
const result = await executeFn(args);
if (isAsyncIterable(result)) {
push({
data: null,
errors: [new GraphQLError('Execution returned AsyncIterable which is not supported!')],
isLive: true,
});
stop();
return;
}
result.isLive = true;
if (finished) {
return;
}
push(result);
}
setTimeout(pump, intervalMs);
}
pump();
stop.then(() => {
finished = true;
});
}));
}
},
};
}