Skip to content

Commit

Permalink
Persisted Query Link: improve memory management
Browse files Browse the repository at this point in the history
  • Loading branch information
phryneas committed Nov 17, 2023
1 parent 9942e61 commit b933dfb
Show file tree
Hide file tree
Showing 4 changed files with 223 additions and 136 deletions.
4 changes: 3 additions & 1 deletion .api-reports/api-report-link_persisted-queries.md
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,9 @@ interface BaseOptions {
// Warning: (ae-forgotten-export) The symbol "ApolloLink" needs to be exported by the entry point index.d.ts
//
// @public (undocumented)
export const createPersistedQueryLink: (options: PersistedQueryLink.Options) => ApolloLink;
export const createPersistedQueryLink: (options: PersistedQueryLink.Options) => ApolloLink & {
resetCache: () => void;
};

// @public (undocumented)
interface DefaultContext extends Record<string, any> {
Expand Down
9 changes: 9 additions & 0 deletions .changeset/thick-tips-cry.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
---
"@apollo/client": patch
---

Persisted Query Link: improve memory management
* use LRU `WeakCache` instead of `WeakMap` to keep a limited number of hash results
* hash cache is initiated lazily, only when needed
* expose `persistedLink.resetHashCache()` method
* reset hash cache if the upstream server reports it doesn't accept persisted queries
61 changes: 61 additions & 0 deletions src/link/persisted-queries/__tests__/persisted-queries.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,32 @@ describe("happy path", () => {
}, reject);
});

it("clears the cache when calling `resetHashCache`", async () => {
fetchMock.post(
"/graphql",
() => new Promise((resolve) => resolve({ body: response })),
{ repeat: 1 }
);

const hashRefs: WeakRef<String>[] = [];
function hash(query: string) {
const newHash = new String(query);
hashRefs.push(new WeakRef(newHash));
return newHash as string;
}
const persistedLink = createPersistedQuery({ sha256: hash });
await new Promise<void>((complete) =>
execute(persistedLink.concat(createHttpLink()), {
query,
variables,
}).subscribe({ complete })
);

await expect(hashRefs[0]).not.toBeGarbageCollected();
persistedLink.resetHashCache();
await expect(hashRefs[0]).toBeGarbageCollected();
});

itAsync("supports loading the hash from other method", (resolve, reject) => {
fetchMock.post(
"/graphql",
Expand Down Expand Up @@ -517,6 +543,41 @@ describe("failure path", () => {
})
);

it.each([
["error message", giveUpResponse],
["error code", giveUpResponseWithCode],
] as const)(
"clears the cache when receiving NotSupported error (%s)",
async (_description, failingResponse) => {
fetchMock.post(
"/graphql",
() => new Promise((resolve) => resolve({ body: failingResponse })),
{ repeat: 1 }
);
fetchMock.post(
"/graphql",
() => new Promise((resolve) => resolve({ body: response })),
{ repeat: 1 }
);

const hashRefs: WeakRef<String>[] = [];
function hash(query: string) {
const newHash = new String(query);
hashRefs.push(new WeakRef(newHash));
return newHash as string;
}
const persistedLink = createPersistedQuery({ sha256: hash });
await new Promise<void>((complete) =>
execute(persistedLink.concat(createHttpLink()), {
query,
variables,
}).subscribe({ complete })
);

await expect(hashRefs[0]).toBeGarbageCollected();
}
);

itAsync("works with multiple errors", (resolve, reject) => {
fetchMock.post(
"/graphql",
Expand Down
Loading

0 comments on commit b933dfb

Please sign in to comment.