@graphql-yoga/plugin-apq
Version:
APQ plugin for GraphQL Yoga.
66 lines (65 loc) • 2.36 kB
JavaScript
import { GraphQLError } from 'graphql';
import { lru } from 'tiny-lru';
export async function hashSHA256(str, api = globalThis) {
const { crypto, TextEncoder } = api;
const textEncoder = new TextEncoder();
const utf8 = textEncoder.encode(str);
const hashBuffer = await crypto.subtle.digest('SHA-256', utf8);
let hashHex = '';
for (const bytes of new Uint8Array(hashBuffer)) {
hashHex += bytes.toString(16).padStart(2, '0');
}
return hashHex;
}
export function createInMemoryAPQStore(options = {}) {
return lru(options.max ?? 1000, options.ttl ?? 36000);
}
function decodeAPQExtension(input) {
if (input != null &&
typeof input === 'object' &&
input?.version === 1 &&
typeof input?.sha256Hash === 'string') {
return input;
}
return null;
}
export function useAPQ(options = {}) {
const { store = createInMemoryAPQStore(), hash = hashSHA256 } = options;
return {
async onParams({ params, setParams, fetchAPI }) {
const persistedQueryData = decodeAPQExtension(params.extensions?.persistedQuery);
if (persistedQueryData === null) {
return;
}
if (params.query == null) {
const persistedQuery = await store.get(persistedQueryData.sha256Hash);
if (persistedQuery == null) {
throw new GraphQLError('PersistedQueryNotFound', {
extensions: {
http: {
status: 404,
},
},
});
}
setParams({
...params,
query: persistedQuery,
});
}
else {
const expectedHash = await hash(params.query, fetchAPI);
if (persistedQueryData.sha256Hash !== expectedHash) {
throw new GraphQLError('PersistedQueryMismatch', {
extensions: {
http: {
status: 400,
},
},
});
}
await store.set(persistedQueryData.sha256Hash, params.query);
}
},
};
}