Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(openapi-fetch): Allow returning Response from onRequest callback #2091

Merged
merged 2 commits into from
Feb 19, 2025

Conversation

p-dubovitsky
Copy link
Contributor

@p-dubovitsky p-dubovitsky commented Jan 4, 2025

Changes

This PR closes #2072 by adding ability to return Response from onRequest middleware hook. This allows middleware to short-circuit the request chain by returning a response directly, which is useful for cases such as deduplicating or caching responses to avoid unnecessary network requests.

How to Review

Please check that:

  • Early Response from onRequest properly skips remaining middleware chain
  • Test coverage is sufficient
  • Documentation clearly explains new functionality

Key implementation details:

  • No changes to existing middleware behavior
  • When Response is returned:
    • The request is not sent to the server
    • Subsequent onRequest handlers are skipped
    • onResponse handlers are skipped

Checklist

  • Unit tests updated
  • docs/ updated (if necessary)
  • pnpm run update:examples run (only applicable for openapi-typescript)

@p-dubovitsky p-dubovitsky requested a review from a team as a code owner January 4, 2025 14:33
@p-dubovitsky p-dubovitsky requested a review from gzm0 January 4, 2025 14:33
Copy link

netlify bot commented Jan 4, 2025

👷 Deploy request for openapi-ts pending review.

Visit the deploys page to approve it

Name Link
🔨 Latest commit 5a38436

Copy link

changeset-bot bot commented Jan 4, 2025

🦋 Changeset detected

Latest commit: 5a38436

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 3 packages
Name Type
openapi-fetch Patch
openapi-react-query Patch
swr-openapi Patch

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

Copy link
Contributor

@gzm0 gzm0 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice! (and sorry for the somewhat tardy review).

Mostly nits on tests, and one doubt in the example that we probably should address.

onResponse({ request, response }) {
if (response.ok) {
const key = getCacheKey(request);
cache.set(key, response);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are you sure we do not need to clone the response here? The response we return is (probably) going to be consumed by the caller.

IIUC, clone() will throw if the response has already been used:

clone() throws a TypeError if the response body has already been used.

(https://developer.mozilla.org/en-US/docs/Web/API/Response/clone)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I think you're right. Thanks for your consideration!

response = result;
break;
} else {
throw new Error("onRequest: must return new Request() or Response() when modifying the request");
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this error message was wrong an now it feels even more wrong :-/ (IIUC you must not return new Request() but mutate the request in place and return it, unclear to me why returning the request is even necessary...).

Probably not blocking for this PR, but a fix would be appreciated.

Copy link
Contributor

@drwpow drwpow Jan 25, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah I’m keen to just remove this (not necessarily in this PR, just in general); this is not true in some cases. It was meant to provide a friendlier error for folks, but I actually think just letting the platform surface the error is more helpful, rather than us getting in the way

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure I understood you correctly. Do you want to throw an error without a message or something else?


test("can return response directly from onRequest", async () => {
const customResponse = Response.json({});
const client = createObservedClient<paths>();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
const client = createObservedClient<paths>();
const client = createObservedClient<paths>({}, () => throw new Error("unexpected call to fetch"));

This way, we make sure we really do not call fetch.


test("skips subsequent onRequest handlers when response is returned", async () => {
let onRequestCalled = false;
const customResponse = Response.json({});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: Inline below? You don't actually need the reference for this test.


test("skips onResponse handlers when response is returned from onRequest", async () => {
let onResponseCalled = false;
const customResponse = Response.json({});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same: Inline below?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice tests!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks!

@duncanbeevers duncanbeevers added the openapi-fetch Relevant to the openapi-fetch library label Feb 15, 2025
@p-dubovitsky p-dubovitsky requested a review from gzm0 February 17, 2025 16:30
@gzm0 gzm0 merged commit ebe56f3 into openapi-ts:main Feb 19, 2025
8 checks passed
@openapi-ts-bot openapi-ts-bot mentioned this pull request Feb 19, 2025
@StrongerMyself
Copy link

StrongerMyself commented Mar 7, 2025

Please update the openapi-fetch package with this functionality. The latest available version in npm is 0.13.4, does not have ‘Early Response’

npm error notarget No matching version found for openapi-fetch@^0.13.5.

https://www.npmjs.com/package/openapi-fetch?activeTab=versions

if (response.ok) {
const key = getCacheKey(request);
cache.set(key, response.clone());
}

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here you must return return response.clone(), otherwise the request body will be read on line 222 in index.js and repeated requests will be rejected with an error

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are you sure? The response in the cache is cloned.

I say this only from looking at the code, if you have tried this and it fails, well, then it fails :)

PR to fix the example in this case is very much appreciated.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am using nextjs 15 (server-side requests). At some point response becomes bodyUsage=true, maybe it is related to the environment - nodesj. Most likely your example works for client-side code, but not for server-side code. I'm not sure how best to flesh out the documentation. In my case, I save data(as text), headers and statuses in Map.

const cache = new Map<string, string>();
const getCacheKey = (request: Request) => `${request.method}:${request.url}`;

const cacheMiddleware: Middleware = {
  onRequest({ request }) {
    const key = getCacheKey(request);
    const cached = cache.get(key);
    if (cached) {
      // Return cached response, skipping actual request and remaining middleware chain
      const response = new Response(cached);
      return response;
    }
  },
  async onResponse({ request, response }) {
    if (response.ok) {
      const key = getCacheKey(request);
      const responseBody = await response.clone().text();
      cache.set(key, responseBody);
    }
  }
};

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

TY for the example. Unfortunately, without more understanding of what is going on here, from the maintainers side, I don't think we have a strong enough signal to take this further.

If you would like, please file a bug with this, and investigation can proceed there. However, please understand that you will be expected to produce a reproducible example that relies on a (standard) fetch implementation only (of course that needs to run in browsers and Node.js, but no SSR magic, just free-standing code).

@openapi-ts-bot openapi-ts-bot mentioned this pull request Mar 14, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
openapi-fetch Relevant to the openapi-fetch library
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Allow returning Response from onRequest callback
5 participants