-
Notifications
You must be signed in to change notification settings - Fork 84
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
Comments
I like the basic API, the one thing I'd like to figure out is how to apply a 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. |
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 🙂 |
Hi @timostamm, do you have any examples using this approach? Thank you very much! |
@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) |
@timostamm I tried it with my code, and MSW is not intercepting the requests. 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! |
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
|
@martines3000, yes, the PR will make this much easier and less error-prone. We'll pick it up again soon. You're using You have two options in this case:
I think this is actually an important point. We have to be very clear that MWS only intercepts |
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 And if compare the request received by Jest version of
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:
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 |
Thanks @timostamm for the quick reply and the solution. It works now 💯 . |
I finally got a chance to play with 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. UtilI originally planned to create a factory function that took a /** @/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 });
});
}; UsageHere 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
|
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:
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
The text was updated successfully, but these errors were encountered: