From 3565dc6d7619a223d0ab0da122cd88062ef6f4e5 Mon Sep 17 00:00:00 2001 From: Gildas Garcia <1122076+djhi@users.noreply.github.com> Date: Thu, 6 Jun 2024 10:59:01 +0200 Subject: [PATCH] Re-organize documentation --- README.md | 381 ++++++++++++++++++++++++------------------- example/fetchMock.ts | 4 +- example/msw.ts | 5 +- example/sinon.ts | 17 +- 4 files changed, 229 insertions(+), 178 deletions(-) diff --git a/README.md b/README.md index 7bef048..2746628 100644 --- a/README.md +++ b/README.md @@ -4,23 +4,23 @@ Intercept AJAX calls to fake a REST server based on JSON data. Use it on top of See it in action in the [react-admin](https://marmelab.com/react-admin/) [demo](https://marmelab.com/react-admin-demo) ([source code](https://github.com/marmelab/react-admin/tree/master/examples/demo)). -## Usage +## Installation ### MSW We recommend you use [MSW](https://mswjs.io/) to mock your API. This will allow you to inspect requests as you usually do in the devtools network tab. -First, install msw and initialize it: +First, install fakerest and MSW. Then initialize MSW: ```sh -npm install msw@latest --save-dev +npm install fakerest msw@latest --save-dev npx msw init # eg: public ``` Then configure it: ```js -// in ./src/msw.js +// in ./src/fakeServer.js import { setupWorker } from "msw/browser"; import { getMswHandlers } from "fakerest"; @@ -41,9 +41,12 @@ const data = { } }; -export const worker = setupWorker(...getMswHandlers({ - data -})); +export const worker = setupWorker( + ...getMswHandlers({ + baseUrl: 'http://localhost:3000', + data + }) +); ``` Finally call the `worker.start()` method before rendering your application. For instance, in a Vite React application: @@ -52,17 +55,17 @@ Finally call the `worker.start()` method before rendering your application. For import React from "react"; import ReactDom from "react-dom"; import { App } from "./App"; -import { worker } from "./msw"; +import { worker } from "./fakeServer"; worker.start().then(() => { ReactDom.render(, document.getElementById("root")); }); ``` -Another option is to use the `MswServer` class. This is useful if you must conditionally include data: +Another option is to use the `MswServer` class. This is useful if you must conditionally include data or add middlewares: ```js -// in ./src/msw.js +// in ./src/fakeServer.js import { setupWorker } from "msw/browser"; import { MswServer } from "fakerest"; @@ -83,19 +86,24 @@ const data = { } }; -const restServer = new MswServer(); -restServer.init(data); +const restServer = new MswServer({ + baseUrl: 'http://localhost:3000', + data, +}); export const worker = setupWorker(...restServer.getHandlers()); ``` +FakeRest will now intercept every `fetch` requests to the REST server. + ### Sinon -```html - - - +const sinonServer = sinon.fakeServer.create(); +// this is required when doing asynchronous XmlHttpRequest +sinonServer.autoRespond = true; + +sinonServer.respondWith( + getSinonHandler({ + baseUrl: 'http://localhost:3000', + data, + }) +); ``` -Another option is to use the `SinonServer` class. This is useful if you must conditionally include data or interceptors: +Another option is to use the `SinonServer` class. This is useful if you must conditionally include data or add middlewares: + +```js +// in ./src/fakeServer.js +import sinon from 'sinon'; +import { SinonServer } from "fakerest"; -```html - - - +const sinonServer = sinon.fakeServer.create(); +// this is required when doing asynchronous XmlHttpRequest +sinonServer.autoRespond = true; + +sinonServer.respondWith( + restServer.getHandler({ + baseUrl: 'http://localhost:3000', + data, + }) +); ``` +FakeRest will now intercept every `XmlHttpRequest` requests to the REST server. + ### fetch-mock +First, install fakerest and fetch-mock: + +```sh +npm install fakerest fetch-mock --save-dev +``` + +You can then initialize the `FetchMockServer`: + ```js +// in ./src/fakeServer.js import fetchMock from 'fetch-mock'; -import FakeRest from 'fakerest'; +import { getFetchMockHandler } from "fakerest"; const data = { 'authors': [ @@ -175,15 +211,15 @@ const data = { fetchMock.mock( 'begin:http://localhost:3000', - FakeRest.getFetchMockHandler({ baseUrl: 'http://localhost:3000', data }) + getFetchMockHandler({ baseUrl: 'http://localhost:3000', data }) ); ``` -Another option is to use the `FetchMockServer` class. This is useful if you must conditionally include data or interceptors: +Another option is to use the `FetchMockServer` class. This is useful if you must conditionally include data or add middlewares: ```js import fetchMock from 'fetch-mock'; -import FakeRest from 'fakerest'; +import { FetchMockServer } from 'fakerest'; const data = { 'authors': [ @@ -201,84 +237,14 @@ const data = { preferred_format: 'hardback', } }; -const restServer = new FakeRest.FetchMockServer({ baseUrl: 'http://localhost:3000' }); -restServer.init(data); +const restServer = new FetchMockServer({ + baseUrl: 'http://localhost:3000', + data +}); fetchMock.mock('begin:http://localhost:3000', restServer.getHandler()); ``` -FakeRest will now intercept every `XmlHttpRequest` to the REST server. The handled routes for collections of items are: - -``` -GET /:resource -POST /:resource -GET /:resource/:id -PUT /:resource/:id -PATCH /:resource/:id -DELETE /:resource/:id -``` - -The handled routes for single items are: - -``` -GET /:resource -PUT /:resource -PATCH /:resource -``` - - -Let's see an example: - -```js -// Query the fake REST server -var req = new XMLHttpRequest(); -req.open("GET", "/authors", false); -req.send(null); -console.log(req.responseText); -// [ -// {"id":0,"first_name":"Leo","last_name":"Tolstoi"}, -// {"id":1,"first_name":"Jane","last_name":"Austen"} -// ] - -var req = new XMLHttpRequest(); -req.open("GET", "/books/3", false); -req.send(null); -console.log(req.responseText); -// {"id":3,"author_id":1,"title":"Sense and Sensibility"} - -var req = new XMLHttpRequest(); -req.open("GET", "/settings", false); -req.send(null); -console.log(req.responseText); -// {"language:"english","preferred_format":"hardback"} - -var req = new XMLHttpRequest(); -req.open("POST", "/books", false); -req.send(JSON.stringify({ author_id: 1, title: 'Emma' })); -console.log(req.responseText); -// {"author_id":1,"title":"Emma","id":4} - -// restore native XHR constructor -server.restore(); -``` - -*Tip*: The `fakerServer` provided by Sinon.js is [available as a standalone library](http://sinonjs.org/docs/#server), without the entire stubbing framework. Simply add the following bower dependency: - -``` -devDependencies: { - "sinon-server": "http://sinonjs.org/releases/sinon-server-1.14.1.js" -} -``` - -## Installation - -FakeRest is available through npm and Bower: - -```sh -# If you use Bower -bower install fakerest --save-dev -# If you use npm -npm install fakerest --save-dev -``` +FakeRest will now intercept every `fetch` requests to the REST server. ## REST Flavor @@ -452,69 +418,146 @@ Operators are specified as suffixes on each filtered field. For instance, applyi GET /books?filter={"price_gte":100} // return books that have a price greater or equal to 100 -## Usage and Configuration +## Middlewares + +All fake servers supports middlewares that allows you to intercept requests and simulate server features such as: + - authentication checks + - server side validation + - server dynamically generated values + - simulate response delays + +A middleware is a function that receive 3 parameters: + - The request object, specific to the chosen mocking solution (e.g. a [`Request`](https://developer.mozilla.org/fr/docs/Web/API/Request) for MSW and `fetch-mock`, a fake [`XMLHttpRequest`](https://developer.mozilla.org/fr/docs/Web/API/XMLHttpRequest) for [Sinon](https://sinonjs.org/releases/v18/fake-xhr-and-server/)) + - The FakeRest context, an object containing the data extracted from the request that FakeRest uses to build the response. It has the following properties: + - `url`: The request URL as a string + - `method`: The request method as a string (`GET`, `POST`, `PATCH` or `PUT`) + - `collection`: The name of the targeted [collection](#collection) (e.g. `posts`) + - `single`: The name of the targeted [single](#single) (e.g. `settings`) + - `requestJson`: The parsed request data if any + - `params`: The request parameters from the URL search (e.g. the identifier of the requested record) + - A function to call the next middleware in the chain + +**Tip**: The middleware function for MSW and `fetch-mock` must return a promise. Those for Sinon must **not** return a promise. + +A middleware must return a FakeRest response either by returning the result of the `next` function or by returning its own response. A FakeRest response is an object with the following properties: + - `status`: The response status as a number (e.g. `200`) + - `headers`: The response HTTP headers as an object where keys are header names + - `body`: The response body which will be stringified + +A middleware might also throw a response specific to the chosen mocking solution (e.g. a [`Response`](https://developer.mozilla.org/fr/docs/Web/API/Response) for MSW, a [`MockResponseObject`](https://www.wheresrhys.co.uk/fetch-mock/#api-mockingmock_response) for `fetch-mock`) for even more control. + +### Authentication Checks + +Here's to implement an authentication check for MSW or `fetch-mock`: ```js -// initialize a rest server with a custom base URL -const restServer = new FakeRest.SinonServer({ baseUrl: 'http://my.custom.domain' }); // only URLs starting with my.custom.domain will be intercepted -restServer.toggleLogging(); // logging is off by default, enable it to see network calls in the console -// Set all JSON data at once - only if identifier name is 'id' -restServer.init(json); -// modify the request before FakeRest handles it, using a request interceptor -// request is { -// url: '...', -// headers: [...], -// requestBody: '...', -// json: ..., // parsed JSON body -// queryString: '...', -// params: {...} // parsed query string -// } -restServer.addRequestInterceptor(function(request) { - var start = (request.params._start - 1) || 0; - var end = request.params._end !== undefined ? (request.params._end - 1) : 19; - request.params.range = [start, end]; - return request; // always return the modified input -}); -// modify the response before FakeRest sends it, using a response interceptor -// response is { -// status: ..., -// headers: [...], -// body: {...} -// } -restServer.addResponseInterceptor(function(response) { - response.body = { data: response.body, status: response.status }; - return response; // always return the modified input +restServer.addMiddleware(async (request, context, next) => { + if (!request.headers?.get('Authorization')) { + throw new Response(null, { status: 401 }); + } + return next(request, context); +} +``` + +Here's how to do the same with Sinon: + +```js +restServer.addMiddleware((request, context, next) => { + if (request.requestHeaders.Authorization === undefined) { + // Uses Sinon API to respond immediately + request.respond(401, {}, 'Unauthorized'); + // Avoid further processing + return null; + } + + return next(request, context); +} +``` + +### Server Side Validation + +Here's to implement server side validation for MSW or `fetch-mock`: + +```js +restServer.addMiddleware(async (request, context, next) => { + if ( + context.collection === 'books' && + context.method === 'POST' && + !context.requestJson?.title + ) { + throw new Response(null, { + status: 400, + statusText: 'Title is required', + }); + } + + return next(request, context); +} +``` + +Here's how to do the same with Sinon: + +```js +restServer.addMiddleware((request, context, next) => { + if ( + context.collection === "books" && + request.method === "POST" && + !context.requestJson?.title + ) { + // Uses Sinon API to respond immediately + request.respond(400, {}, "Title is required"); + // Avoid further processing + return null; + } + + return next(request, context); +} +``` + +### Server Dynamically Generated Values + +Here's to implement server dynamically generated values: + +```js +restServer.addMiddleware(async (request, context, next) => { + if ( + context.collection === 'books' && + context.method === 'POST' + ) { + const response = await next(request, context); + response.body.updatedAt = new Date().toISOString(); + return response; + } + + return next(request, context); +} +``` + +### Simulate Response Delays + +This only works with MSW and `fetch-mock`: + +```js +restServer.addMiddleware(async (request, context, next) => { + return new Promise((resolve) => { + setTimeout(() => { + resolve(next(request, context)); + }, delayMs); + }); }); -// set default query, e.g. to force embeds or filters -restServer.setDefaultQuery(function(resourceName) { - if (resourceName == 'authors') return { embed: ['books'] } - if (resourceName == 'books') return { filter: { published: true } } - return {}; -}) -// enable batch request handler, i.e. allow API clients to query several resources into a single request -// see [Facebook's Batch Requests philosophy](https://developers.facebook.com/docs/graph-api/making-multiple-requests) for more details. -restServer.setBatchUrl('/batch'); - -// you can create more than one fake server to listen to several domains -const restServer2 = new FakeRest.SinonServer({ baseUrl: 'http://my.other.domain' }); -// Set data collection by collection - allows to customize the identifier name -const authorsCollection = new FakeRest.Collection({ items: [], identifierName: '_id' }); -authorsCollection.addOne({ first_name: 'Leo', last_name: 'Tolstoi' }); // { _id: 0, first_name: 'Leo', last_name: 'Tolstoi' } -authorsCollection.addOne({ first_name: 'Jane', last_name: 'Austen' }); // { _id: 1, first_name: 'Jane', last_name: 'Austen' } -// collections have auto incremented identifiers by default but accept identifiers already set -authorsCollection.addOne({ _id: 3, first_name: 'Marcel', last_name: 'Proust' }); // { _id: 3, first_name: 'Marcel', last_name: 'Proust' } -restServer2.addCollection('authors', authorsCollection); -// collections are mutable -authorsCollection.updateOne(1, { last_name: 'Doe' }); // { _id: 1, first_name: 'Jane', last_name: 'Doe' } -authorsCollection.removeOne(3); // { _id: 3, first_name: 'Marcel', last_name: 'Proust' } - -const server = sinon.fakeServer.create(); -server.autoRespond = true; -server.respondWith(restServer.getHandler()); -server.respondWith(restServer2.getHandler()); ``` -## Configure Identifiers Generation +This is so common FakeRest provides the `withDelay` function for that: + +```js +import { withDelay } from 'fakerest'; + +restServer.addMiddleware(withDelay(300)); +``` + +## Configuration + +### Configure Identifiers Generation By default, FakeRest uses an auto incremented sequence for the items identifiers. If you'd rather use UUIDs for instance but would like to avoid providing them when you insert new items, you can provide your own function: diff --git a/example/fetchMock.ts b/example/fetchMock.ts index dae14ab..89935b0 100644 --- a/example/fetchMock.ts +++ b/example/fetchMock.ts @@ -1,10 +1,10 @@ import fetchMock from 'fetch-mock'; -import { FetchServer, withDelay } from 'fakerest'; +import { FetchMockServer, withDelay } from 'fakerest'; import { data } from './data'; import { dataProvider as defaultDataProvider } from './dataProvider'; export const initializeFetchMock = () => { - const restServer = new FetchServer({ + const restServer = new FetchMockServer({ baseUrl: 'http://localhost:3000', data, loggingEnabled: true, diff --git a/example/msw.ts b/example/msw.ts index 4f1ffd2..60f04c3 100644 --- a/example/msw.ts +++ b/example/msw.ts @@ -1,5 +1,4 @@ import { setupWorker } from 'msw/browser'; -import { HttpResponse } from 'msw'; import { MswServer, withDelay } from '../src/FakeRest'; import { data } from './data'; import { dataProvider as defaultDataProvider } from './dataProvider'; @@ -12,7 +11,7 @@ const restServer = new MswServer({ restServer.addMiddleware(withDelay(300)); restServer.addMiddleware(async (request, context, next) => { if (!request.headers?.get('Authorization')) { - throw new HttpResponse(null, { status: 401 }); + throw new Response(null, { status: 401 }); } if ( @@ -20,7 +19,7 @@ restServer.addMiddleware(async (request, context, next) => { request.method === 'POST' && !context.requestJson?.title ) { - throw new HttpResponse(null, { + throw new Response(null, { status: 400, statusText: 'Title is required', }); diff --git a/example/sinon.ts b/example/sinon.ts index e08f620..5eb3b35 100644 --- a/example/sinon.ts +++ b/example/sinon.ts @@ -16,20 +16,29 @@ export const initializeSinon = () => { return null; } + if ( + context.collection === 'books' && + request.method === 'POST' && + !context.requestJson?.title + ) { + request.respond(400, {}, 'Title is required'); + return null; + } + return next(request, context); }); // use sinon.js to monkey-patch XmlHttpRequest - const server = sinon.fakeServer.create(); + const sinonServer = sinon.fakeServer.create(); // this is required when doing asynchronous XmlHttpRequest - server.autoRespond = true; + sinonServer.autoRespond = true; if (window) { // @ts-ignore window.restServer = restServer; // give way to update data in the console // @ts-ignore - window.sinonServer = server; // give way to update data in the console + window.sinonServer = sinonServer; // give way to update data in the console } - server.respondWith(restServer.getHandler()); + sinonServer.respondWith(restServer.getHandler()); }; export const dataProvider: DataProvider = {