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

Support API mocking via MSW #825

Open
timostamm opened this issue Sep 15, 2023 · 10 comments
Open

Support API mocking via MSW #825

timostamm opened this issue Sep 15, 2023 · 10 comments
Labels
enhancement New feature or request

Comments

@timostamm
Copy link
Member

Is your feature request related to a problem? Please describe.
When writing a web application that makes many API calls, it is often challenging to get good code coverage without mocking the API.

Describe the solution you'd like
The Mock Service Worker library can intercept requests on the network level and mock responses in a framework-agnostic way, and it runs on Node.js too.

MSW can already be used with Connect, but it requires mocking on the HTTP level, without any benefits that the schema provides: For example, it is easy to misspell a header name or request path - even though both are already well-defined by the schema or protocol.

It would be fantastic if there was a type-safe integration for MSW with Connect to remove the boilerplate, for example:

import { setupWorker } from 'msw'
import { service } from '@connectrpc/connect-msw'

const worker = setupWorker(
  service(ElizaService, {
    say(req) {
      return { sentence: `you said ${req.sentence}` }
    }
  }),
)

worker.start()

Describe alternatives you've considered
@connectrpc/connect-playwright provides API mocking with Playwright, but that means all tests must be written in Playwright, limiting the options.

Additional context

@timostamm timostamm added the enhancement New feature or request label Sep 15, 2023
@paul-sachs
Copy link
Contributor

paul-sachs commented Sep 15, 2023

I like the basic API, the one thing I'd like to figure out is how to apply a baseUrl. Ideally we'd only have to define it once (in the transport) but there's no real good way to extract that. Actually that would probably be a bad idea anyways since services could have diff baseUrls anyways. So I think the second arg might need to be an options object:

const worker = setupWorker(
  ...service(ElizaService, { baseUrl: "/api" }, {
    say(req) {
      return { sentence: `you said ${req.sentence}` }
    }
  }),
)

Though it's not quite as attractive an api, and it gets worse with multiple services:

const worker = setupWorker(
  ...service(ElizaService, { baseUrl: "/api" }, {
    say(req) {
      return { sentence: `you said ${req.sentence}` }
    }
  }),
  ...service(BigIntService, { baseUrl: "/api" }, {
    add() {
      return { result: 2n };
    }
  )
)

Though maybe that's fine. I thought of pushing the options arg to the last, but the service impl will always be the most important (and largest) argument so I feel like it shouldn't be stuck in the center of argument list.

@timostamm
Copy link
Member Author

I should have been more clear that the code snippet was really just a pseudo-code example to illustrate that users don't have to hand-write requests paths and serialization code - a pretty nice benefit from the schema.

We'll certainly need to accept several options (the baseUrl you mention, but also serialization options, and likely also server side interceptors from #527). I'm sure that a reasonable API will manifest, even if it might need a couple of iterations 🙂

@nguyenyou
Copy link
Contributor

nguyenyou commented Oct 22, 2023

MSW can already be used with Connect, but it requires mocking on the HTTP level

Hi @timostamm, do you have any examples using this approach? Thank you very much!

@timostamm
Copy link
Member Author

@nguyenyou, the Connect protocol is very simple, especially for unary and JSON. This should mock the rpc Say of the demo service Eliza:

import { http, HttpResponse } from 'msw'
 
export const handlers = [
  http.post('connectrpc.eliza.v1.ElizaService/Say', async (request) => {
    const requestJson = await request.json();
    return HttpResponse.json({
      sentence: `you said: ${requestJson.sentence}`
    })
  }),
]

(Based on the current examples on mswjs.io)

@martines3000
Copy link

martines3000 commented Nov 8, 2023

@timostamm
Hi. I have a question regarding the proposed solution you provided.

I tried it with my code, and MSW is not intercepting the requests.
Could this maybe be related to me using MSW@2.0.3 ?

This is how I call my RPC method:

const transport = createConnectTransport({
  baseUrl: `http://127.0.0.1:3003/grpc`,
  httpVersion: '1.1',
});
const client = createPromiseClient(ExampleService, transport);
return client.hello({ name: 'bob' });

This is how I tried mocking it (I tried all possible URL combinations, because I thought the problem was there, but now I'm not sure anymore):

const mswServer = setupServer();
mswServer.listen({ onUnhandledRequest: 'error' });

const handler = http.post(
  'http://127.0.0.1:3003/grpc/martines3000.example.v1.ExampleService/Hello',
  // eslint-disable-next-line @typescript-eslint/require-await
  async (request) => {
    console.log('request', request);
    return HttpResponse.arrayBuffer(
      new HelloResponse({ message: 'Does not work' }).toBinary(),
      { status: 200 },
    );
  },
);

mswServer.use(handler);

The handler code is a little bit different, but the problem is that it never gets to this part (MSW never intercepts the request).

Do you maybe have any ideas where I should look ?

The PR #830 implementation will probably solve all of this issues and making mocking much easier, right ?

Thanks for the help!

@boan-anbo
Copy link

boan-anbo commented Nov 8, 2023

@timostamm Hi. I have a question regarding the proposed solution you provided.

@martines3000

I got it working with MSW 2.0.3 using the #830 PR's util method like this in Jest test set up.

import { setupServer } from 'msw/node'

export const handlersTest = service(
   UnitKeywordService,
   {
   	baseUrl: 'http://127.0.0.1:23012',
   },
   {
   	querySomething: () => {
   		return {
   			something: new SampleSomething(),
   		}
   	},
   }
)

export const mockServer = setupServer(...handlersTest)

beforeAll(() => mockServer.listen())

You can find the service method here:

export function service<T extends ServiceType>(

@timostamm
Copy link
Member Author

@martines3000, yes, the PR will make this much easier and less error-prone. We'll pick it up again soon.

You're using createConnectTransport from @connectrpc/connect-node, which is using the Node.js built-in modules http and http2. msw can only intercept requests that use fetch() for HTTP requests.

You have two options in this case:

  1. Use createConnectTransport from @connectrpc/connect-web instead for your clients. It will use fetch() for HTTP requests, and that works very well on recent versions of Node.js.
  2. Continue to use createConnectTransport from @connectrpc/connect-node, but use the router transport for testing instead of msw. It's documented here.

I think this is actually an important point. We have to be very clear that MWS only intercepts fetch(). That's not self-explanatory just from reading the MSW docs or this issue description.

@boan-anbo
Copy link

boan-anbo commented Nov 8, 2023

@timostamm Hi. I have a question regarding the proposed solution you provided.

@martines3000

I got it working with MSW 2.0.3 using the #830 PR's util method like this in Jest test set up.

import { setupServer } from 'msw/node'

export const handlersTest = service(
   UnitKeywordService,
   {
   	baseUrl: 'http://127.0.0.1:23012',
   },
   {
   	querySomething: () => {
   		return {
   			something: new SampleSomething(),
   		}
   	},
   }
)

export const mockServer = setupServer(...handlersTest)

beforeAll(() => mockServer.listen())

You can find the service method here:

export function service<T extends ServiceType>(

However, I do have a related issue using the above method, which completely confuses me.

When I use the exact same setup in Storybook with its msw add-on

mswjs/msw-storybook-addon#121 (comment)

The mocked call errs with a ConnectError","message":"[deadline_exceeded] the operation timed out by the browser worker.

And if compare the request received by Jest version of createResponseResolver and the one received by storybook msw-addon setupWorker createResponseResolver, the timeout header are different:

  • With default timeout ms = 12345m
export const grpcWebTransport = createGrpcWebTransport({
	baseUrl: 'http://127.0.0.1:23012',
	useBinaryFormat: true,
	defaultTimeoutMs: 12345,
})
const headerGrpcTimeout = request.headers.get('grpc-timeout')
console.log('headerGrpcTimeout', headerGrpcTimeout)

the Jest resolver receives:

12345m

while the storybook msw-addon setup receives:

() => date.getTime() - Date.now()m`. 

// or

() => void m

which causes the timeout error (?)

I don't know why this happens. Cause MSW add-on doesn't seem to do anything different, and both are using the exact same client and unary method call. But somehow one's GrpcTiemout header is incorrect.

@martines3000
Copy link

Thanks @timostamm for the quick reply and the solution. It works now 💯 .
Also thanks @boan-anbo for the suggestion, will try it out.

@is-jonreeves
Copy link

I finally got a chance to play with v2 of protobuf-es and was looking about to see if there was any information on using msw with it. I stumbled across this thread and the #830 PR and it helped me solve a few issues I was running up against. Would love to see an official @connectrpc/connect-msw at somepoint!

I'm not sure if its the correct way to go about it (as I suspect information about the transport/configuration should probably be taken into account), but figured I'd share what I ended up with as it seemed to work ok in my small example project.

Util

I originally planned to create a factory function that took a baseUrl and transport and returned a another curried factory that could then be used to create handlers for that endpoint (using just the service, method and response). I didn't get that far though and also didn't see a way to avoid having to pass in the schema too.

/** @/test/createStubConnectHandler.ts */

import type { DescService, MessageInitShape, MessageShape } from '@bufbuild/protobuf';
import type { GenMessage } from '@bufbuild/protobuf/codegenv1';
import { ConnectError, createConnectRouter, type ServiceImpl } from '@connectrpc/connect';
import { http, HttpHandler, HttpResponse } from 'msw';

// Types
type ServiceMethod<S> = S extends DescService ? keyof S['method'] : never;
type ResponseSchema<S, M extends ServiceMethod<S>> = S extends DescService ? GenMessage<MessageShape<S['method'][M]['output']>> : never;
type OutputMessage<S, M extends ServiceMethod<S>> = S extends DescService ? MessageInitShape<S['method'][M]['output']> : never;

type StubConnectHandlerFactory = <
  S extends DescService,
  M extends ServiceMethod<S>,
  T extends ResponseSchema<S, M>,
  O extends OutputMessage<S, M>,
>(options: { baseUrl: string, service: S, method: M, schema: T, response: O | ConnectError }) => HttpHandler;

// Helpers
const AsyncIterator = {
  fromStream: async function* <T>(stream?: ReadableStream<T> | null): AsyncGenerator<T> {
    if (stream === undefined || stream === null) return;
    const reader = stream.getReader();
    try {
      while (true) {
        const { done, value } = await reader.read();
        if (done) return;
        yield value;
      }
    } finally {
      reader.releaseLock();
    }
  },
  toStream: function <T>(input?: AsyncIterable<T>): ReadableStream<T> | undefined {
    if (input === undefined || input === null) return;
    const stream = new ReadableStream<T>({
      async start(controller) {
        try {
          for await (const chunk of input) controller.enqueue(chunk);
        } finally {
          controller.close();
        }
      },
    });
    return stream;
  },
};

// Factory
export const createStubConnectHandler: StubConnectHandlerFactory = (options) => {
  // Build URL
  const endpoint = `${options.baseUrl}/${options.service.typeName}/${options.method.toString()}`;

  // Create Mock Handler
  return http.post(endpoint, async ({ request }) => {
    // Copy Request Parts
    const { url, body: bodyStream, method, headers: header, signal } = request.clone();

    // Create Body AsyncIterator
    const body = AsyncIterator.fromStream(bodyStream);

    // Create RPC Handler
    const handler = createConnectRouter()
      .service(options.service, {
        [options.method]: async () => {
          if (options.response instanceof ConnectError) throw options.response;
          return options.response;
        },
      } as Partial<ServiceImpl<any>>)
      .handlers[0];

    // Run RPC Handler
    const result = await handler({ httpVersion: '2.0', url, method, header, body, signal });

    // Respond
    return new HttpResponse(AsyncIterator.toStream(result.body), { status: result.status, headers: result.header });
  });
};

Usage

Here are some example unary response handlers showing both success and failure. Its pretty clean to work with and the type-inference/guarding is really nice.

/** @/test/handlers.ts */

import { Code, ConnectError } from '@connectrpc/connect';
import { SystemService, GetSystemInfoResponseDtoSchema } from '@example/sdk';

import { createStubConnectHandler } from '@/test/createStubConnectHandler';

export const getSystemInfoHandlerSuccess = createStubConnectHandler({
  baseUrl: 'http://localhost:8080',
  service: SystemService,
  method: 'getSystemInfo',
  schema: GetSystemInfoResponseDtoSchema,
  response: {
    version: '1.0.0',
  },
});

export const getSystemInfoHandlerFailure = createStubConnectHandler({
  baseUrl: 'http://localhost:8080',
  service: SystemService,
  method: 'getSystemInfo',
  schema: GetSystemInfoResponseDtoSchema,
  response: new ConnectError('Not Authenticated!', Code.Unauthenticated),
});

Configuring msw and vitest

I was using @bufbuild/protobuf@2.2.3, @connectrpc/connect@2.0.0, @connectrpc/connect-web@2.0.0, msw@2.7.0, vite@6.0.3, vitest@2.1.8 and configuration and test cases looked a like this...

show code
/** @/test/server.ts */

import { setupServer } from 'msw/node';

import { getSystemInfoHandlerSuccess } from '@/test/handlers';

const handlers = [
  getSystemInfoHandlerSuccess,
];

export const server = setupServer(...handlers);
/** @/test/setup.ts */

import { server } from '@/test/server';

beforeAll(() => {
  server.listen({ onUnhandledRequest: 'error' });
});

afterEach(() => {
  server.resetHandlers();
});

afterAll(() => {
  server.close();
});
/** @/src/example.test.ts */

...
import { server } from '@/test/server';
import { getSystemInfoHandlerFailure } from '@/test/handlers';
...

it('should return system info when authenticated', async () => {
  // Setup
  const request = create(GetSystemInfoRequestDtoSchema, {});
  const expected = create(GetSystemInfoResponseDtoSchema, { version: '1.0.0' });

  // Test
  const actual = await client.getSystemInfo(request);

  // Check
  expect(actual).toStrictEqual(expected);
});

it('should throw an error if not authenticated', async () => {
  // Setup
  const request = create(GetSystemInfoRequestDtoSchema, {});
  const expected = new ConnectError('Not Authenticated!', Code.Unauthenticated);
  server.use(getSystemInfoHandlerFailure);

  // Test
  const actual = client.getSystemInfo(request);

  // Check
  await expect(actual).rejects.toThrow(expected);
});

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging a pull request may close this issue.

6 participants