├── CLAUDE.md ├── .gitignore ├── .prettierrc ├── .husky └── pre-commit ├── example ├── vite-env.d.ts ├── data.ts ├── dataProvider.ts ├── fetchMock.ts ├── msw.ts ├── middlewares.ts ├── index.tsx ├── authProvider.ts ├── sinon.ts ├── App.tsx └── users.json ├── src ├── withDelay.ts ├── index.ts ├── parseQueryString.ts ├── types.ts ├── adapters │ ├── MswAdapter.ts │ ├── FetchMockAdapter.ts │ ├── SinonAdapter.ts │ └── SinonAdapter.spec.ts ├── Single.ts ├── Database.ts ├── Database.spec.ts ├── Single.spec.ts ├── SimpleRestServer.spec.ts ├── SimpleRestServer.ts └── Collection.ts ├── tsconfig.json ├── Makefile ├── biome.json ├── LICENSE ├── vite.config.min.ts ├── vite.config.ts ├── .github └── workflows │ └── ci.yml ├── package.json ├── UPGRADE.md ├── AGENTS.md ├── index.html ├── public └── mockServiceWorker.js └── README.md /CLAUDE.md: -------------------------------------------------------------------------------- 1 | @AGENTS.md -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | dist 3 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "tabWidth": 4, 3 | "singleQuote": true 4 | } -------------------------------------------------------------------------------- /.husky/pre-commit: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | . "$(dirname "$0")/_/husky.sh" 3 | 4 | npx lint-staged -------------------------------------------------------------------------------- /example/vite-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | 3 | interface ImportMetaEnv { 4 | readonly VITE_MOCK: 'msw' | 'fetch-mock' | 'sinon'; 5 | } 6 | 7 | interface ImportMeta { 8 | readonly env: ImportMetaEnv; 9 | } 10 | -------------------------------------------------------------------------------- /src/withDelay.ts: -------------------------------------------------------------------------------- 1 | import type { Middleware } from './SimpleRestServer.ts'; 2 | 3 | export const withDelay = 4 | (delayMs: number): Middleware => 5 | (context, next) => { 6 | return new Promise((resolve) => { 7 | setTimeout(() => { 8 | resolve(next(context)); 9 | }, delayMs); 10 | }); 11 | }; 12 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | export * from './types.ts'; 2 | export * from './adapters/SinonAdapter.ts'; 3 | export * from './adapters/FetchMockAdapter.ts'; 4 | export * from './adapters/MswAdapter.ts'; 5 | export * from './Database.ts'; 6 | export * from './SimpleRestServer.ts'; 7 | export * from './Collection.ts'; 8 | export * from './Single.ts'; 9 | export * from './withDelay.ts'; 10 | -------------------------------------------------------------------------------- /example/data.ts: -------------------------------------------------------------------------------- 1 | export const data = { 2 | authors: [ 3 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 4 | { id: 1, first_name: 'Jane', last_name: 'Austen' }, 5 | ], 6 | books: [ 7 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 8 | { id: 1, author_id: 0, title: 'War and Peace' }, 9 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 10 | { id: 3, author_id: 1, title: 'Sense and Sensibility' }, 11 | ], 12 | settings: { 13 | language: 'english', 14 | preferred_format: 'hardback', 15 | }, 16 | }; 17 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "rootDir": "src", 4 | "outDir": "./dist", 5 | "noEmit": true, 6 | "declaration": true, 7 | "strict": true, 8 | "esModuleInterop": true, 9 | "module": "node16", 10 | "moduleResolution": "node16", 11 | "moduleDetection": "force", 12 | "target": "ES2022", // Node.js 18 13 | "types": ["vitest/globals"], 14 | "jsx": "react-jsx", 15 | "allowImportingTsExtensions": true, 16 | "skipLibCheck": true 17 | }, 18 | "include": ["src"], 19 | "exclude": ["node_modules/**"] 20 | } 21 | -------------------------------------------------------------------------------- /Makefile: -------------------------------------------------------------------------------- 1 | .PHONY: build test 2 | 3 | install: 4 | @npm install 5 | 6 | build-dev: 7 | @NODE_ENV=development npm run build 8 | 9 | build: 10 | @NODE_ENV=production npm run build 11 | 12 | run: run-msw 13 | 14 | run-msw: 15 | @NODE_ENV=development VITE_MOCK=msw npm run dev 16 | 17 | run-fetch-mock: 18 | @NODE_ENV=development VITE_MOCK=fetch-mock npm run dev 19 | 20 | run-sinon: 21 | @NODE_ENV=development VITE_MOCK=sinon npm run dev 22 | 23 | watch: 24 | @NODE_ENV=development npm run build --watch 25 | 26 | test: 27 | @npm run test 28 | 29 | format: 30 | @npm run format 31 | 32 | lint: 33 | @npm run lint -------------------------------------------------------------------------------- /biome.json: -------------------------------------------------------------------------------- 1 | { 2 | "$schema": "https://biomejs.dev/schemas/1.7.0/schema.json", 3 | "organizeImports": { 4 | "enabled": true 5 | }, 6 | "formatter": { 7 | "enabled": true, 8 | "indentStyle": "space", 9 | "indentWidth": 4, 10 | "ignore": ["public/*.js"] 11 | }, 12 | "javascript": { 13 | "formatter": { 14 | "quoteStyle": "single" 15 | } 16 | }, 17 | "linter": { 18 | "enabled": true, 19 | "rules": { 20 | "recommended": true, 21 | "suspicious": { 22 | "noExplicitAny": "off" 23 | } 24 | }, 25 | "ignore": ["public/*.js"] 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /example/dataProvider.ts: -------------------------------------------------------------------------------- 1 | import simpleRestProvider from 'ra-data-simple-rest'; 2 | import { fetchUtils } from 'react-admin'; 3 | 4 | const httpClient = (url: string, options: any = {}) => { 5 | if (!options.headers) { 6 | options.headers = new Headers({ Accept: 'application/json' }); 7 | } 8 | const persistedUser = localStorage.getItem('user'); 9 | const user = persistedUser ? JSON.parse(persistedUser) : null; 10 | if (user) { 11 | options.headers.set('Authorization', `Bearer ${user.id}`); 12 | } 13 | return fetchUtils.fetchJson(url, options); 14 | }; 15 | 16 | export const dataProvider = simpleRestProvider( 17 | 'http://localhost:3000', 18 | httpClient, 19 | ); 20 | -------------------------------------------------------------------------------- /src/parseQueryString.ts: -------------------------------------------------------------------------------- 1 | export function parseQueryString(queryString: string) { 2 | if (!queryString) { 3 | return {}; 4 | } 5 | const queryObject: Record = {}; 6 | const queryElements = queryString.split('&'); 7 | 8 | queryElements.map((queryElement) => { 9 | if (queryElement.indexOf('=') === -1) { 10 | queryObject[queryElement] = true; 11 | } else { 12 | let [key, value] = queryElement.split('='); 13 | if (value.indexOf('[') === 0 || value.indexOf('{') === 0) { 14 | value = JSON.parse(value); 15 | } 16 | queryObject[key.trim()] = value; 17 | } 18 | }); 19 | 20 | return queryObject; 21 | } 22 | -------------------------------------------------------------------------------- /example/fetchMock.ts: -------------------------------------------------------------------------------- 1 | import fetchMock from 'fetch-mock'; 2 | import { FetchMockAdapter } from '../src'; 3 | import { data } from './data'; 4 | import { dataProvider as defaultDataProvider } from './dataProvider'; 5 | import { middlewares } from './middlewares'; 6 | 7 | export const initializeFetchMock = () => { 8 | const restServer = new FetchMockAdapter({ 9 | baseUrl: 'http://localhost:3000', 10 | data, 11 | loggingEnabled: true, 12 | middlewares, 13 | }); 14 | if (window) { 15 | // @ts-ignore 16 | window.restServer = restServer; // give way to update data in the console 17 | } 18 | 19 | fetchMock.mock('begin:http://localhost:3000', restServer.getHandler()); 20 | }; 21 | 22 | export const dataProvider = defaultDataProvider; 23 | -------------------------------------------------------------------------------- /example/msw.ts: -------------------------------------------------------------------------------- 1 | import { http } from 'msw'; 2 | import { setupWorker } from 'msw/browser'; 3 | import { getMswHandler } from '../src'; 4 | import { data } from './data'; 5 | import { dataProvider as defaultDataProvider } from './dataProvider'; 6 | import { middlewares } from './middlewares'; 7 | 8 | export const initializeMsw = async () => { 9 | const handler = getMswHandler({ 10 | baseUrl: 'http://localhost:3000', 11 | data, 12 | middlewares, 13 | }); 14 | const worker = setupWorker(http.all(/http:\/\/localhost:3000/, handler)); 15 | return worker.start({ 16 | quiet: true, // Instruct MSW to not log requests in the console 17 | onUnhandledRequest: 'bypass', // Instruct MSW to ignore requests we don't handle 18 | }); 19 | }; 20 | 21 | export const dataProvider = defaultDataProvider; 22 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2015 marmelab 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy 4 | of this software and associated documentation files (the "Software"), to deal 5 | in the Software without restriction, including without limitation the rights 6 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 7 | copies of the Software, and to permit persons to whom the Software is 8 | furnished to do so, subject to the following conditions: 9 | 10 | The above copyright notice and this permission notice shall be included in 11 | all copies or substantial portions of the Software. 12 | 13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 14 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 15 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 16 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 17 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 18 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 19 | THE SOFTWARE. 20 | -------------------------------------------------------------------------------- /vite.config.min.ts: -------------------------------------------------------------------------------- 1 | /// 2 | import { resolve } from 'node:path'; 3 | import { defineConfig } from 'vite'; 4 | 5 | export default defineConfig({ 6 | build: { 7 | lib: { 8 | // Could also be a dictionary or array of multiple entry points 9 | entry: resolve(__dirname, 'src/index.ts'), 10 | name: 'FakeRest', 11 | // the proper extensions will be added 12 | fileName: 'fakerest.min', 13 | }, 14 | minify: true, 15 | sourcemap: true, 16 | emptyOutDir: false, 17 | rollupOptions: { 18 | // make sure to externalize deps that shouldn't be bundled 19 | // into your library 20 | external: ['lodash'], 21 | output: { 22 | // Provide global variables to use in the UMD build 23 | // for externalized deps 24 | globals: { 25 | lodash: '_', 26 | }, 27 | }, 28 | }, 29 | }, 30 | test: { 31 | globals: true, 32 | environment: 'happy-dom', 33 | }, 34 | }); 35 | -------------------------------------------------------------------------------- /example/middlewares.ts: -------------------------------------------------------------------------------- 1 | import { withDelay } from '../src'; 2 | import { data } from './data'; 3 | 4 | export const middlewares = [ 5 | withDelay(300), 6 | async (context, next) => { 7 | if (!context.headers?.get('Authorization')) { 8 | return { 9 | status: 401, 10 | headers: {}, 11 | }; 12 | } 13 | return next(context); 14 | }, 15 | async (context, next) => { 16 | if (context.collection === 'books' && context.method === 'POST') { 17 | if ( 18 | data[context.collection].some( 19 | (book) => book.title === context.requestBody?.title, 20 | ) 21 | ) { 22 | return { 23 | body: { 24 | errors: { 25 | title: 'An article with this title already exists. The title must be unique.', 26 | }, 27 | }, 28 | status: 400, 29 | headers: {}, 30 | }; 31 | } 32 | } 33 | return next(context); 34 | }, 35 | ]; 36 | -------------------------------------------------------------------------------- /example/index.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { createRoot } from 'react-dom/client'; 3 | import { App } from './App'; 4 | 5 | const root = createRoot(document.getElementById('root')); 6 | 7 | switch (import.meta.env.VITE_MOCK) { 8 | case 'fetch-mock': 9 | import('./fetchMock') 10 | .then(({ initializeFetchMock, dataProvider }) => { 11 | initializeFetchMock(); 12 | return dataProvider; 13 | }) 14 | .then((dataProvider) => { 15 | root.render(); 16 | }); 17 | break; 18 | case 'sinon': 19 | import('./sinon') 20 | .then(({ initializeSinon, dataProvider }) => { 21 | initializeSinon(); 22 | return dataProvider; 23 | }) 24 | .then((dataProvider) => { 25 | root.render(); 26 | }); 27 | break; 28 | default: 29 | import('./msw') 30 | .then(({ initializeMsw, dataProvider }) => { 31 | return initializeMsw().then(() => dataProvider); 32 | }) 33 | .then((dataProvider) => { 34 | root.render(); 35 | }); 36 | } 37 | -------------------------------------------------------------------------------- /vite.config.ts: -------------------------------------------------------------------------------- 1 | /// 2 | import { resolve } from 'node:path'; 3 | import { defineConfig } from 'vite'; 4 | import dts from 'vite-plugin-dts'; 5 | import react from '@vitejs/plugin-react'; 6 | 7 | export default defineConfig({ 8 | build: { 9 | lib: { 10 | // Could also be a dictionary or array of multiple entry points 11 | entry: resolve(__dirname, 'src/index.ts'), 12 | name: 'FakeRest', 13 | // the proper extensions will be added 14 | fileName: 'fakerest', 15 | }, 16 | minify: false, 17 | sourcemap: true, 18 | rollupOptions: { 19 | // make sure to externalize deps that shouldn't be bundled 20 | // into your library 21 | external: ['lodash'], 22 | output: { 23 | // Provide global variables to use in the UMD build 24 | // for externalized deps 25 | globals: { 26 | lodash: '_', 27 | }, 28 | }, 29 | }, 30 | }, 31 | plugins: [react(), dts()], 32 | test: { 33 | globals: true, 34 | environment: 'happy-dom', 35 | }, 36 | resolve: { 37 | alias: { 38 | fakerest: resolve(__dirname, 'src/FakeRest.ts'), 39 | }, 40 | }, 41 | }); 42 | -------------------------------------------------------------------------------- /.github/workflows/ci.yml: -------------------------------------------------------------------------------- 1 | name: "Tests" 2 | 3 | on: 4 | push: 5 | paths-ignore: 6 | - 'docs/**' 7 | branches: 8 | - master 9 | - next 10 | pull_request: 11 | paths-ignore: 12 | - 'docs/**' 13 | 14 | jobs: 15 | lint: 16 | runs-on: ubuntu-latest 17 | steps: 18 | - name: Checkout 19 | uses: actions/checkout@v2 20 | - name: Use Node.js LTS 21 | uses: actions/setup-node@v1 22 | with: 23 | node-version: '18.x' 24 | - uses: bahmutov/npm-install@v1 25 | - name: Lint Check 26 | run: npx @biomejs/biome lint src 27 | - name: Format Check 28 | run: npx @biomejs/biome format src 29 | typecheck: 30 | runs-on: ubuntu-latest 31 | steps: 32 | - name: Checkout 33 | uses: actions/checkout@v2 34 | - name: Use Node.js LTS 35 | uses: actions/setup-node@v1 36 | with: 37 | node-version: '18.x' 38 | - uses: bahmutov/npm-install@v1 39 | - name: Typecheck 40 | run: npx tsc 41 | unit-test: 42 | runs-on: ubuntu-latest 43 | steps: 44 | - name: Checkout 45 | uses: actions/checkout@v2 46 | - name: Use Node.js LTS 47 | uses: actions/setup-node@v1 48 | with: 49 | node-version: '18.x' 50 | - uses: bahmutov/npm-install@v1 51 | - name: Unit Tests 52 | run: make test 53 | env: 54 | CI: true 55 | -------------------------------------------------------------------------------- /src/types.ts: -------------------------------------------------------------------------------- 1 | export type CollectionItem = { [key: string]: any }; 2 | 3 | export type SortFunction = ( 4 | a: T, 5 | b: T, 6 | ) => number; 7 | export type Sort = string | [string, 'asc' | 'desc'] | SortFunction; 8 | 9 | export type Range = [number, number] | [number]; 10 | 11 | export type FilterFunction = ( 12 | item: T, 13 | ) => boolean; 14 | export type FilterObject = CollectionItem & { q?: string }; 15 | export type Filter = FilterObject | FilterFunction; 16 | 17 | export type Query = { 18 | filter?: Filter; 19 | sort?: Sort; 20 | range?: Range; 21 | embed?: Embed; 22 | }; 23 | 24 | export type QueryFunction = (name: string) => Query; 25 | 26 | export type Predicate = ( 27 | item: T, 28 | ) => boolean; 29 | 30 | export type Embed = string | string[]; 31 | 32 | export type BaseResponse = { 33 | status: number; 34 | body?: Record | Record[]; 35 | headers: { [key: string]: string }; 36 | }; 37 | 38 | export type FakeRestContext = { 39 | url?: string; 40 | headers?: Headers; 41 | method?: string; 42 | collection?: string; 43 | single?: string; 44 | requestBody: Record | undefined; 45 | params: { [key: string]: any }; 46 | }; 47 | 48 | export type NormalizedRequest = Pick< 49 | FakeRestContext, 50 | 'url' | 'method' | 'params' | 'requestBody' | 'headers' 51 | >; 52 | 53 | export type APIServer = { 54 | baseUrl?: string; 55 | handle: (context: FakeRestContext) => Promise; 56 | }; 57 | -------------------------------------------------------------------------------- /example/authProvider.ts: -------------------------------------------------------------------------------- 1 | import { type AuthProvider, HttpError } from 'react-admin'; 2 | import data from './users.json'; 3 | 4 | /** 5 | * This authProvider is only for test purposes. Don't use it in production. 6 | */ 7 | export const authProvider: AuthProvider = { 8 | login: ({ username, password }) => { 9 | const user = data.users.find( 10 | (u) => u.username === username && u.password === password, 11 | ); 12 | 13 | if (user) { 14 | const { password, ...userToPersist } = user; 15 | localStorage.setItem('user', JSON.stringify(userToPersist)); 16 | return Promise.resolve(); 17 | } 18 | 19 | return Promise.reject( 20 | new HttpError('Unauthorized', 401, { 21 | message: 'Invalid username or password', 22 | }), 23 | ); 24 | }, 25 | logout: () => { 26 | localStorage.removeItem('user'); 27 | return Promise.resolve(); 28 | }, 29 | checkError: (error) => { 30 | const status = error.status; 31 | if (status === 401 || status === 403) { 32 | localStorage.removeItem('auth'); 33 | return Promise.reject(); 34 | } 35 | // other error code (404, 500, etc): no need to log out 36 | return Promise.resolve(); 37 | }, 38 | checkAuth: () => 39 | localStorage.getItem('user') ? Promise.resolve() : Promise.reject(), 40 | getPermissions: () => { 41 | return Promise.resolve(undefined); 42 | }, 43 | getIdentity: () => { 44 | const persistedUser = localStorage.getItem('user'); 45 | const user = persistedUser ? JSON.parse(persistedUser) : null; 46 | 47 | return Promise.resolve(user); 48 | }, 49 | }; 50 | 51 | export default authProvider; 52 | -------------------------------------------------------------------------------- /src/adapters/MswAdapter.ts: -------------------------------------------------------------------------------- 1 | import { SimpleRestServer } from '../SimpleRestServer.ts'; 2 | import type { BaseServerOptions } from '../SimpleRestServer.ts'; 3 | import type { APIServer, NormalizedRequest } from '../types.ts'; 4 | 5 | export class MswAdapter { 6 | server: APIServer; 7 | 8 | constructor({ server, ...options }: MswAdapterOptions) { 9 | this.server = server || new SimpleRestServer(options); 10 | } 11 | 12 | getHandler() { 13 | return async ({ request }: { request: Request }) => { 14 | const normalizedRequest = await this.getNormalizedRequest(request); 15 | const response = await this.server.handle(normalizedRequest); 16 | return new Response(JSON.stringify(response.body), { 17 | status: response.status, 18 | headers: response.headers, 19 | }); 20 | }; 21 | } 22 | 23 | async getNormalizedRequest(request: Request): Promise { 24 | const url = new URL(request.url); 25 | const params = Object.fromEntries( 26 | Array.from(new URLSearchParams(url.search).entries()).map( 27 | ([key, value]) => [key, JSON.parse(value)], 28 | ), 29 | ); 30 | let requestBody: Record | undefined = undefined; 31 | try { 32 | const text = await request.text(); 33 | requestBody = JSON.parse(text); 34 | } catch (e) { 35 | // not JSON, no big deal 36 | } 37 | 38 | return { 39 | url: request.url, 40 | headers: request.headers, 41 | params, 42 | requestBody, 43 | method: request.method, 44 | }; 45 | } 46 | } 47 | 48 | export const getMswHandler = (options: MswAdapterOptions) => { 49 | const server = new MswAdapter(options); 50 | return server.getHandler(); 51 | }; 52 | 53 | export type MswAdapterOptions = BaseServerOptions & { 54 | server?: APIServer; 55 | }; 56 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "fakerest", 3 | "version": "4.2.0", 4 | "repository": "https://github.com/marmelab/FakeRest", 5 | "description": "Patch XMLHttpRequest to fake a REST server based on JSON data. ", 6 | "scripts": { 7 | "dev": "vite", 8 | "build": "vite build && vite build -c vite.config.min.ts", 9 | "format": "biome format --write src", 10 | "lint": "biome lint --apply src", 11 | "test": "vitest", 12 | "prepare": "husky" 13 | }, 14 | "type": "module", 15 | "main": "dist/fakerest.min.umd.cjs", 16 | "module": "./dist/fakerest.min.js", 17 | "types": "./dist/index.d.ts", 18 | "exports": { 19 | ".": { 20 | "types": "./dist/index.d.ts", 21 | "import": "./dist/fakerest.min.js", 22 | "require": "./dist/fakerest.umd.cjs" 23 | } 24 | }, 25 | "author": "François Zaninotto ", 26 | "license": "MIT", 27 | "devDependencies": { 28 | "@biomejs/biome": "1.7.0", 29 | "@types/lodash": "^4.17.0", 30 | "@types/sinon": "^17.0.3", 31 | "@vitejs/plugin-react": "^5.1.2", 32 | "fetch-mock": "^9.11.0", 33 | "happy-dom": "^20.0.0", 34 | "husky": "^9.0.11", 35 | "lint-staged": "^15.2.2", 36 | "msw": "^2.2.14", 37 | "ra-data-simple-rest": "^5.11.2", 38 | "react": "^18.0.0", 39 | "react-admin": "^5.11.2", 40 | "react-dom": "^18.0.0", 41 | "sinon": "~18.0.0", 42 | "typescript": "^5.4.5", 43 | "vite": "^7.3.0", 44 | "vite-plugin-dts": "^4.5.4", 45 | "vitest": "^4.0.15" 46 | }, 47 | "dependencies": { 48 | "lodash": "^4.17.21" 49 | }, 50 | "browserslist": "> 0.25%, not dead", 51 | "lint-staged": { 52 | "*.{js,jsx,ts,tsx}": [ 53 | "biome lint --apply", 54 | "biome format --write" 55 | ] 56 | }, 57 | "msw": { 58 | "workerDirectory": [ 59 | "public" 60 | ] 61 | } 62 | } 63 | -------------------------------------------------------------------------------- /example/sinon.ts: -------------------------------------------------------------------------------- 1 | import sinon from 'sinon'; 2 | import simpleRestProvider from 'ra-data-simple-rest'; 3 | import { HttpError, type Options } from 'react-admin'; 4 | import { SinonAdapter } from '../src'; 5 | import { data } from './data'; 6 | import { middlewares } from './middlewares'; 7 | 8 | export const initializeSinon = () => { 9 | const restServer = new SinonAdapter({ 10 | baseUrl: 'http://localhost:3000', 11 | data, 12 | loggingEnabled: true, 13 | middlewares, 14 | }); 15 | 16 | // use sinon.js to monkey-patch XmlHttpRequest 17 | const server = sinon.fakeServer.create(); 18 | // this is required when doing asynchronous XmlHttpRequest 19 | server.autoRespond = true; 20 | if (window) { 21 | // @ts-ignore 22 | window.restServer = restServer; // give way to update data in the console 23 | // @ts-ignore 24 | window.sinonServer = server; // give way to update data in the console 25 | } 26 | server.respondWith(restServer.getHandler()); 27 | }; 28 | 29 | // An HttpClient based on XMLHttpRequest to use with Sinon 30 | const httpClient = (url: string, options: Options = {}): Promise => { 31 | const request = new XMLHttpRequest(); 32 | request.open(options.method ?? 'GET', url); 33 | 34 | const persistedUser = localStorage.getItem('user'); 35 | const user = persistedUser ? JSON.parse(persistedUser) : null; 36 | if (user) { 37 | request.setRequestHeader('Authorization', `Bearer ${user.id}`); 38 | } 39 | 40 | // add content-type header 41 | request.overrideMimeType('application/json'); 42 | request.send(typeof options.body === 'string' ? options.body : undefined); 43 | 44 | return new Promise((resolve, reject) => { 45 | request.onloadend = (e) => { 46 | let json: any; 47 | try { 48 | json = JSON.parse(request.responseText); 49 | } catch (e) { 50 | // not json, no big deal 51 | } 52 | // Get the raw header string 53 | const headersAsString = request.getAllResponseHeaders(); 54 | 55 | // Convert the header string into an array 56 | // of individual headers 57 | const arr = headersAsString.trim().split(/[\r\n]+/); 58 | 59 | // Create a map of header names to values 60 | const headers = new Headers(); 61 | for (const line of arr) { 62 | const parts = line.split(': '); 63 | const header = parts.shift(); 64 | if (!header) continue; 65 | const value = parts.join(': '); 66 | headers.set(header, value); 67 | } 68 | if (request.status < 200 || request.status >= 300) { 69 | return reject( 70 | new HttpError( 71 | json?.message || request.statusText, 72 | request.status, 73 | json, 74 | ), 75 | ); 76 | } 77 | resolve({ 78 | status: request.status, 79 | headers, 80 | body: request.responseText, 81 | json, 82 | }); 83 | }; 84 | }); 85 | }; 86 | 87 | export const dataProvider = simpleRestProvider( 88 | 'http://localhost:3000', 89 | httpClient, 90 | ); 91 | -------------------------------------------------------------------------------- /example/App.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | import { 3 | Admin, 4 | Create, 5 | type DataProvider, 6 | EditGuesser, 7 | Resource, 8 | ShowGuesser, 9 | required, 10 | AutocompleteInput, 11 | ReferenceInput, 12 | SimpleForm, 13 | TextInput, 14 | SearchInput, 15 | Datagrid, 16 | List, 17 | TextField, 18 | FunctionField, 19 | } from 'react-admin'; 20 | 21 | import authProvider from './authProvider'; 22 | import { QueryClient } from '@tanstack/react-query'; 23 | 24 | const queryClient = new QueryClient({ 25 | defaultOptions: { 26 | queries: { 27 | refetchOnWindowFocus: false, 28 | }, 29 | }, 30 | }); 31 | 32 | export const App = ({ dataProvider }: { dataProvider: DataProvider }) => { 33 | return ( 34 | 39 | 46 | 52 | `${record.first_name} ${record.last_name}` 53 | } 54 | /> 55 | 56 | ); 57 | }; 58 | 59 | const AuthorList = () => ( 60 | ]}> 61 | 62 | 63 | 64 | 65 | 66 | 67 | ); 68 | 69 | // The default value for the title field should cause a server validation error as it's not unique 70 | const BookCreate = () => ( 71 | 72 | 73 | 74 | 75 | 76 | 81 | 82 | 83 | ); 84 | 85 | const bookFilters = [ 86 | , 87 | , 92 | ]; 93 | 94 | const BookList = () => ( 95 | 96 | 97 | 98 | 102 | record.author 103 | ? `${record.author.first_name} ${record.author.last_name}` 104 | : '' 105 | } 106 | /> 107 | 108 | 109 | 110 | ); 111 | -------------------------------------------------------------------------------- /UPGRADE.md: -------------------------------------------------------------------------------- 1 | # Upgrading to 4.0.0 2 | 3 | ## Dropped bower support 4 | 5 | Fakerest no longer supports bower. You can still use it in your project by installing it via npm: 6 | 7 | ```bash 8 | npm install fakerest 9 | ``` 10 | 11 | ## Renamed `Server` to `SinonAdapter` 12 | 13 | The `Server` class has been renamed to `SinonAdapter` and now expects a configuration object instead of a URL. 14 | 15 | ```diff 16 | import sinon from 'sinon'; 17 | -import { Server } from 'fakerest'; 18 | +import { SinonAdapter } from 'fakerest'; 19 | import { data } from './data'; 20 | 21 | -const server = new Server('http://myapi.com'); 22 | -server.init(data); 23 | +const server = new SinonAdapter({ baseUrl: 'http://myapi.com', data }); 24 | const server = sinon.fakeServer.create(); 25 | server.respondWith(server.getHandler()); 26 | ``` 27 | 28 | ## Renamed `FetchServer` to `FetchMockAdapter` 29 | 30 | The `FetchServer` class has been renamed to `FetchMockAdapter` and now expects a configuration object instead of a URL. 31 | 32 | ```diff 33 | import fetchMock from 'fetch-mock'; 34 | -import { FetchServer } from 'fakerest'; 35 | +import { FetchMockAdapter } from 'fakerest'; 36 | import { data } from './data'; 37 | 38 | -const server = new FetchServer('http://myapi.com'); 39 | -server.init(data); 40 | +const server = new FetchMockAdapter({ baseUrl: 'http://myapi.com', data }); 41 | fetchMock.mock('begin:http://myapi.com', server.getHandler()); 42 | ``` 43 | 44 | ## Constructor Of `Collection` Takes An Object 45 | 46 | ```diff 47 | import { Collection } from 'fakerest'; 48 | 49 | -const posts = new Collection([ 50 | - { id: 1, title: 'baz' }, 51 | - { id: 2, title: 'biz' }, 52 | - { id: 3, title: 'boz' }, 53 | -]); 54 | +const posts = new Collection({ 55 | + items: [ 56 | + { id: 1, title: 'baz' }, 57 | + { id: 2, title: 'biz' }, 58 | + { id: 3, title: 'boz' }, 59 | + ], 60 | +}); 61 | ``` 62 | 63 | ## `addCollection` is now `adapter.server.addCollection` 64 | 65 | ```diff 66 | import fetchMock from 'fetch-mock'; 67 | -import { FetchServer } from 'fakerest'; 68 | +import { FetchMockAdapter } from 'fakerest'; 69 | import { posts } from './posts'; 70 | 71 | -const server = new FetchServer('http://myapi.com'); 72 | -server.addCollection('posts', posts); 73 | -fetchMock.mock('begin:http://myapi.com', server.getHandler()); 74 | +const adapter = new FetchMockAdapter({ baseUrl: 'http://myapi.com', data }); 75 | +adapter.server.addCollection('posts', posts); 76 | +fetchMock.mock('begin:http://myapi.com', adapter.getHandler()); 77 | ``` 78 | 79 | ## Request and Response Interceptors Have Been Replaced By Middlewares 80 | 81 | Fakerest used to have request and response interceptors. We replaced those with middlewares. They allow much more use cases. 82 | 83 | Migrate your request interceptors to middlewares passed when building the handler: 84 | 85 | ```diff 86 | - const myRequestInterceptor = function(request) { 87 | + const myMiddleware = async function(context, next) { 88 | var start = (request.params._start - 1) || 0; 89 | var end = request.params._end !== undefined ? (request.params._end - 1) : 19; 90 | request.params.range = [start, end]; 91 | - return request; // always return the modified input 92 | + return next(context); 93 | }; 94 | 95 | -restServer.addRequestInterceptor(myRequestInterceptor); 96 | +const handler = new getMswHandler({ 97 | + baseUrl: 'http://my.custom.domain', 98 | + data, 99 | + middlewares: [myMiddleware], 100 | }); 101 | ``` 102 | 103 | Migrate your response interceptors the same way. 104 | -------------------------------------------------------------------------------- /src/Single.ts: -------------------------------------------------------------------------------- 1 | import cloneDeep from 'lodash/cloneDeep.js'; 2 | import type { Database } from './Database.ts'; 3 | import type { CollectionItem, Embed, Query } from './types.ts'; 4 | 5 | export class Single { 6 | obj: T | null = null; 7 | database: Database | null = null; 8 | name: string | null = null; 9 | 10 | constructor(obj: T) { 11 | if (!(obj instanceof Object)) { 12 | throw new Error( 13 | "Can't initialize a Single with anything except an object", 14 | ); 15 | } 16 | this.obj = cloneDeep(obj); 17 | } 18 | 19 | /** 20 | * A Single may need to access other collections (e.g. for embedded 21 | * references) This is done through a reference to the parent database. 22 | */ 23 | setDatabase(database: Database) { 24 | this.database = database; 25 | } 26 | 27 | setName(name: string) { 28 | this.name = name; 29 | } 30 | 31 | // No need to embed Singles, since they are by their nature top-level 32 | // No need to worry about remote references, (i.e. mysingleton_id=1) since 33 | // it is by definition a singleton 34 | _oneToManyEmbedder(resourceName: string) { 35 | return (item: T) => { 36 | if (this.database == null) { 37 | throw new Error("Can't embed references without a database"); 38 | } 39 | const otherCollection = this.database.collections[resourceName]; 40 | if (!otherCollection) 41 | throw new Error( 42 | `Can't embed a non-existing collection ${resourceName}`, 43 | ); 44 | // We have an array of ids {posts: [1,2]} (back refs are not valid 45 | // for singleton) 46 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 47 | item[resourceName] = otherCollection.getAll({ 48 | filter: (i: CollectionItem) => 49 | item[resourceName].indexOf( 50 | i[otherCollection.identifierName], 51 | ) !== -1, 52 | }); 53 | return item; 54 | }; 55 | } 56 | 57 | _manyToOneEmbedder(resourceName: string) { 58 | const pluralResourceName = `${resourceName}s`; 59 | const referenceName = `${resourceName}_id`; 60 | return (item: T) => { 61 | if (this.database == null) { 62 | throw new Error("Can't embed references without a database"); 63 | } 64 | const otherCollection = 65 | this.database.collections[pluralResourceName]; 66 | if (!otherCollection) 67 | throw new Error( 68 | `Can't embed a non-existing collection ${resourceName}`, 69 | ); 70 | try { 71 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 72 | item[resourceName] = otherCollection.getOne( 73 | item[referenceName], 74 | ); 75 | } catch (e) { 76 | // Resource doesn't exist, so don't embed 77 | } 78 | return item; 79 | }; 80 | } 81 | 82 | _itemEmbedder(embed: Embed) { 83 | const resourceNames = Array.isArray(embed) ? embed : [embed]; 84 | const resourceEmbedders = resourceNames.map((resourceName) => 85 | resourceName.endsWith('s') 86 | ? this._oneToManyEmbedder(resourceName) 87 | : this._manyToOneEmbedder(resourceName), 88 | ); 89 | return (item: T) => 90 | resourceEmbedders.reduce( 91 | (itemWithEmbeds, embedder) => embedder(itemWithEmbeds), 92 | item, 93 | ); 94 | } 95 | 96 | getOnly(query?: Query) { 97 | let item = this.obj; 98 | if (query?.embed && this.database) { 99 | item = Object.assign({}, item); // Clone 100 | item = this._itemEmbedder(query.embed)(item); 101 | } 102 | return item; 103 | } 104 | 105 | updateOnly(item: T) { 106 | if (this.obj == null) { 107 | throw new Error("Can't update a non-existing object"); 108 | } 109 | 110 | for (const key in item) { 111 | this.obj[key] = item[key]; 112 | } 113 | return this.obj; 114 | } 115 | } 116 | -------------------------------------------------------------------------------- /src/adapters/FetchMockAdapter.ts: -------------------------------------------------------------------------------- 1 | import { SimpleRestServer } from '../SimpleRestServer.ts'; 2 | import { parseQueryString } from '../parseQueryString.ts'; 3 | import type { BaseServerOptions } from '../SimpleRestServer.ts'; 4 | import type { BaseResponse, APIServer, NormalizedRequest } from '../types.ts'; 5 | import type { MockResponseObject } from 'fetch-mock'; 6 | 7 | export class FetchMockAdapter { 8 | loggingEnabled = false; 9 | server: APIServer; 10 | 11 | constructor({ 12 | loggingEnabled = false, 13 | server, 14 | ...options 15 | }: FetchMockAdapterOptions = {}) { 16 | this.server = server || new SimpleRestServer(options); 17 | this.loggingEnabled = loggingEnabled; 18 | } 19 | 20 | getHandler() { 21 | const handler = async (url: string, options: RequestInit) => { 22 | const request = new Request(url, options); 23 | const normalizedRequest = await this.getNormalizedRequest(request); 24 | const response = await this.server.handle(normalizedRequest); 25 | this.log(request, response, normalizedRequest); 26 | return response as MockResponseObject; 27 | }; 28 | 29 | return handler; 30 | } 31 | 32 | async getNormalizedRequest(request: Request): Promise { 33 | const req = 34 | typeof request === 'string' ? new Request(request) : request; 35 | const queryString = req.url 36 | ? decodeURIComponent(req.url.slice(req.url.indexOf('?') + 1)) 37 | : ''; 38 | const params = parseQueryString(queryString); 39 | const text = await req.text(); 40 | let requestBody: Record | undefined = undefined; 41 | try { 42 | requestBody = JSON.parse(text); 43 | } catch (e) { 44 | // not JSON, no big deal 45 | } 46 | 47 | return { 48 | url: req.url, 49 | headers: req.headers, 50 | params, 51 | requestBody, 52 | method: req.method, 53 | }; 54 | } 55 | 56 | log( 57 | request: FetchMockFakeRestRequest, 58 | response: BaseResponse, 59 | normalizedRequest: NormalizedRequest, 60 | ) { 61 | if (!this.loggingEnabled) return; 62 | if (console.group) { 63 | // Better logging in Chrome 64 | console.groupCollapsed( 65 | normalizedRequest.method, 66 | normalizedRequest.url, 67 | '(FakeRest)', 68 | ); 69 | console.group('request'); 70 | console.log(normalizedRequest.method, normalizedRequest.url); 71 | console.log('headers', request.headers); 72 | console.log('body ', request.requestJson); 73 | console.groupEnd(); 74 | console.group('response', response.status); 75 | console.log('headers', response.headers); 76 | console.log('body ', response.body); 77 | console.groupEnd(); 78 | console.groupEnd(); 79 | } else { 80 | console.log( 81 | 'FakeRest request ', 82 | normalizedRequest.method, 83 | normalizedRequest.url, 84 | 'headers', 85 | request.headers, 86 | 'body', 87 | request.requestJson, 88 | ); 89 | console.log( 90 | 'FakeRest response', 91 | response.status, 92 | 'headers', 93 | response.headers, 94 | 'body', 95 | response.body, 96 | ); 97 | } 98 | } 99 | 100 | toggleLogging() { 101 | this.loggingEnabled = !this.loggingEnabled; 102 | } 103 | } 104 | 105 | export const getFetchMockHandler = (options: FetchMockAdapterOptions) => { 106 | const server = new FetchMockAdapter(options); 107 | return server.getHandler(); 108 | }; 109 | 110 | /** 111 | * @deprecated Use FetchServer instead 112 | */ 113 | export const FetchServer = FetchMockAdapter; 114 | 115 | export type FetchMockFakeRestRequest = Partial & { 116 | requestBody?: string; 117 | responseText?: string; 118 | requestJson?: Record; 119 | queryString?: string; 120 | params?: { [key: string]: any }; 121 | }; 122 | 123 | export type FetchMockAdapterOptions = BaseServerOptions & { 124 | server?: APIServer; 125 | loggingEnabled?: boolean; 126 | }; 127 | -------------------------------------------------------------------------------- /src/Database.ts: -------------------------------------------------------------------------------- 1 | import { Collection } from './Collection.ts'; 2 | import { Single } from './Single.ts'; 3 | import type { CollectionItem, Query, QueryFunction } from './types.ts'; 4 | 5 | export class Database { 6 | identifierName = 'id'; 7 | collections: Record> = {}; 8 | singles: Record> = {}; 9 | getNewId?: () => number | string; 10 | 11 | constructor({ 12 | data, 13 | identifierName = 'id', 14 | getNewId, 15 | }: DatabaseOptions = {}) { 16 | this.getNewId = getNewId; 17 | this.identifierName = identifierName; 18 | 19 | if (data) { 20 | this.init(data); 21 | } 22 | } 23 | 24 | /** 25 | * Shortcut for adding several collections if identifierName is always the same 26 | */ 27 | init(data: Record) { 28 | for (const name in data) { 29 | const value = data[name]; 30 | if (Array.isArray(value)) { 31 | this.addCollection( 32 | name, 33 | new Collection({ 34 | items: value, 35 | identifierName: this.identifierName, 36 | getNewId: this.getNewId, 37 | }), 38 | ); 39 | } else { 40 | this.addSingle(name, new Single(value)); 41 | } 42 | } 43 | } 44 | 45 | addCollection( 46 | name: string, 47 | collection: Collection, 48 | ) { 49 | this.collections[name] = collection; 50 | collection.setDatabase(this); 51 | collection.setName(name); 52 | } 53 | 54 | getCollection(name: string) { 55 | return this.collections[name]; 56 | } 57 | 58 | getCollectionNames() { 59 | return Object.keys(this.collections); 60 | } 61 | 62 | addSingle( 63 | name: string, 64 | single: Single, 65 | ) { 66 | this.singles[name] = single; 67 | single.setDatabase(this); 68 | single.setName(name); 69 | } 70 | 71 | getSingle(name: string) { 72 | return this.singles[name]; 73 | } 74 | 75 | getSingleNames() { 76 | return Object.keys(this.singles); 77 | } 78 | 79 | /** 80 | * @param {string} name 81 | * @param {string} params As decoded from the query string, e.g. { sort: "name", filter: {enabled:true}, slice: [10, 20] } 82 | */ 83 | getCount(name: string, params?: Query) { 84 | return this.collections[name].getCount(params); 85 | } 86 | 87 | /** 88 | * @param {string} name 89 | * @param {string} params As decoded from the query string, e.g. { sort: "name", filter: {enabled:true}, slice: [10, 20] } 90 | */ 91 | getAll(name: string, params?: Query) { 92 | return this.collections[name].getAll(params); 93 | } 94 | 95 | getOne(name: string, identifier: string | number, params?: Query) { 96 | return this.collections[name].getOne(identifier, params); 97 | } 98 | 99 | addOne(name: string, item: CollectionItem) { 100 | if (!Object.prototype.hasOwnProperty.call(this.collections, name)) { 101 | this.addCollection( 102 | name, 103 | new Collection({ 104 | items: [], 105 | identifierName: this.identifierName, 106 | getNewId: this.getNewId, 107 | }), 108 | ); 109 | } 110 | return this.collections[name].addOne(item); 111 | } 112 | 113 | updateOne(name: string, identifier: string | number, item: CollectionItem) { 114 | return this.collections[name].updateOne(identifier, item); 115 | } 116 | 117 | removeOne(name: string, identifier: string | number) { 118 | return this.collections[name].removeOne(identifier); 119 | } 120 | 121 | getOnly(name: string, params?: Query) { 122 | return this.singles[name].getOnly(); 123 | } 124 | 125 | updateOnly(name: string, item: CollectionItem) { 126 | return this.singles[name].updateOnly(item); 127 | } 128 | } 129 | 130 | export type DatabaseOptions = { 131 | baseUrl?: string; 132 | batchUrl?: string | null; 133 | data?: Record; 134 | defaultQuery?: QueryFunction; 135 | identifierName?: string; 136 | getNewId?: () => number | string; 137 | loggingEnabled?: boolean; 138 | }; 139 | -------------------------------------------------------------------------------- /AGENTS.md: -------------------------------------------------------------------------------- 1 | # CLAUDE.md 2 | 3 | This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository. 4 | 5 | ## Project Overview 6 | 7 | FakeRest is a browser library that intercepts AJAX calls to mock a REST server based on JSON data. It provides adapters for MSW, fetch-mock, and Sinon.js to enable testing of JavaScript REST clients without a backend server. 8 | 9 | ## Common Commands 10 | 11 | ### Development 12 | ```bash 13 | npm run dev # Run dev server (uses MSW by default) 14 | make run-msw # Run with MSW adapter 15 | make run-fetch-mock # Run with fetch-mock adapter 16 | make run-sinon # Run with Sinon adapter 17 | ``` 18 | 19 | ### Testing & Quality 20 | ```bash 21 | npm test # Run tests with Vitest 22 | npm run format # Format code with Biome 23 | npm run lint # Lint code with Biome 24 | ``` 25 | 26 | ### Building 27 | ```bash 28 | npm run build # Build both minified and non-minified versions 29 | make build # Production build via Make 30 | ``` 31 | 32 | ### Running Single Tests 33 | ```bash 34 | npx vitest run [test-file-pattern] # Run specific test file 35 | npx vitest [test-name-pattern] # Run tests matching pattern 36 | ``` 37 | 38 | ## Architecture 39 | 40 | ### Core Components 41 | 42 | The library has a layered architecture: 43 | 44 | 1. **Adapters Layer** (`src/adapters/`) 45 | - `MswAdapter`: Integrates with MSW (Mock Service Worker) 46 | - `FetchMockAdapter`: Integrates with fetch-mock 47 | - `SinonAdapter`: Integrates with Sinon.js fake server 48 | - Each adapter normalizes requests from their respective mocking library and transforms responses back 49 | 50 | 2. **Server Layer** (`SimpleRestServer`) 51 | - Implements REST semantics (GET, POST, PUT, PATCH, DELETE) 52 | - Handles routing to collections vs singles 53 | - Processes middleware chain 54 | - URL pattern: `/{collection}` or `/{collection}/{id}` or `/{single}` 55 | 56 | 3. **Database Layer** (`Database`) 57 | - Manages collections (arrays of records) and singles (single objects) 58 | - Routes CRUD operations to appropriate collection/single 59 | - Handles initialization from data objects 60 | 61 | 4. **Collection/Single Layer** 62 | - `Collection`: Implements filtering, sorting, pagination, and embedding for array data 63 | - `Single`: Manages single object resources (e.g., user profile, settings) 64 | - Both support embedding related resources 65 | 66 | ### Request Flow 67 | 68 | ``` 69 | Mocking Library (MSW/fetch-mock/Sinon) 70 | ↓ 71 | Adapter (normalizes request) 72 | ↓ 73 | SimpleRestServer.handle() 74 | ↓ 75 | Middleware chain (optional) 76 | ↓ 77 | SimpleRestServer.handleRequest() 78 | ↓ 79 | Database → Collection/Single 80 | ↓ 81 | Response (normalized) 82 | ↓ 83 | Adapter (transforms to library format) 84 | ↓ 85 | Mocking Library 86 | ``` 87 | 88 | ### Key Concepts 89 | 90 | - **Collections**: Array-based resources that support filtering (including `q` for full-text search, operators like `_gte`, `_lte`, `_eq`, `_neq`), sorting, range queries, and embedding 91 | - **Singles**: Single object resources (not arrays) for endpoints like `/settings` or `/me` 92 | - **Embedding**: Automatically resolve relationships by embedding related collections/singles in responses via `embed` parameter 93 | - **Middleware**: Functions that intercept requests to add authentication, validation, delays, or dynamic values 94 | - **Identifiers**: Customizable per collection (default: `id`, common alternative: `_id` for MongoDB-style APIs) 95 | 96 | ### File Structure 97 | 98 | ``` 99 | src/ 100 | ├── adapters/ # Adapter implementations for different mocking libraries 101 | ├── Collection.ts # Collection logic (filtering, sorting, pagination) 102 | ├── Database.ts # Database managing collections and singles 103 | ├── SimpleRestServer.ts # REST server implementation with middleware support 104 | ├── Single.ts # Single object resource logic 105 | ├── types.ts # TypeScript type definitions 106 | ├── withDelay.ts # Middleware helper for simulating delays 107 | ├── parseQueryString.ts # Query parameter parsing 108 | └── index.ts # Main entry point, exports public API 109 | ``` 110 | 111 | ## Code Style 112 | 113 | - **Formatter/Linter**: Biome (configured in `biome.json`) 114 | - 4-space indentation 115 | - Single quotes for strings 116 | - Explicit any types allowed (`noExplicitAny: off`) 117 | - **TypeScript**: All source files use `.ts` extension with explicit `.ts` imports 118 | - **Testing**: Vitest with happy-dom environment for DOM emulation 119 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 9 | 10 | 11 | 12 | FakeRest Demo 13 | 109 | 110 | 114 | 115 | 116 | 117 | 118 | 119 | 120 | 121 | 122 |
123 |
124 |
Loading...
125 |
126 |
127 | 128 | 129 | -------------------------------------------------------------------------------- /src/Database.spec.ts: -------------------------------------------------------------------------------- 1 | import { Database } from './Database.ts'; 2 | import { Single } from './Single.ts'; 3 | import { Collection } from './Collection.ts'; 4 | 5 | describe('Database', () => { 6 | describe('init', () => { 7 | it('should populate several collections', () => { 8 | const server = new Database(); 9 | server.init({ 10 | foo: [{ a: 1 }, { a: 2 }, { a: 3 }], 11 | bar: [{ b: true }, { b: false }], 12 | baz: { name: 'baz' }, 13 | }); 14 | expect(server.getAll('foo')).toEqual([ 15 | { id: 0, a: 1 }, 16 | { id: 1, a: 2 }, 17 | { id: 2, a: 3 }, 18 | ]); 19 | expect(server.getAll('bar')).toEqual([ 20 | { id: 0, b: true }, 21 | { id: 1, b: false }, 22 | ]); 23 | expect(server.getOnly('baz')).toEqual({ name: 'baz' }); 24 | }); 25 | }); 26 | 27 | describe('addCollection', () => { 28 | it('should add a collection and index it by name', () => { 29 | const server = new Database(); 30 | const collection = new Collection({ 31 | items: [ 32 | { id: 1, name: 'foo' }, 33 | { id: 2, name: 'bar' }, 34 | ], 35 | }); 36 | server.addCollection('foo', collection); 37 | const newcollection = server.getCollection('foo'); 38 | expect(newcollection).toEqual(collection); 39 | }); 40 | }); 41 | 42 | describe('addSingle', () => { 43 | it('should add a single object and index it by name', () => { 44 | const server = new Database(); 45 | const single = new Single({ name: 'foo', description: 'bar' }); 46 | server.addSingle('foo', single); 47 | expect(server.getSingle('foo')).toEqual(single); 48 | }); 49 | }); 50 | 51 | describe('getAll', () => { 52 | it('should return all items for a given name', () => { 53 | const server = new Database(); 54 | server.addCollection( 55 | 'foo', 56 | new Collection({ 57 | items: [ 58 | { id: 1, name: 'foo' }, 59 | { id: 2, name: 'bar' }, 60 | ], 61 | }), 62 | ); 63 | server.addCollection( 64 | 'baz', 65 | new Collection({ items: [{ id: 1, name: 'baz' }] }), 66 | ); 67 | expect(server.getAll('foo')).toEqual([ 68 | { id: 1, name: 'foo' }, 69 | { id: 2, name: 'bar' }, 70 | ]); 71 | expect(server.getAll('baz')).toEqual([{ id: 1, name: 'baz' }]); 72 | }); 73 | 74 | it('should support a query', () => { 75 | const server = new Database(); 76 | server.addCollection( 77 | 'foo', 78 | new Collection({ 79 | items: [ 80 | { id: 0, name: 'c', arg: false }, 81 | { id: 1, name: 'b', arg: true }, 82 | { id: 2, name: 'a', arg: true }, 83 | ], 84 | }), 85 | ); 86 | const params = { 87 | filter: { arg: true }, 88 | sort: 'name', 89 | slice: [0, 10], 90 | }; 91 | const expected = [ 92 | { id: 2, name: 'a', arg: true }, 93 | { id: 1, name: 'b', arg: true }, 94 | ]; 95 | expect(server.getAll('foo', params)).toEqual(expected); 96 | }); 97 | }); 98 | 99 | describe('getOne', () => { 100 | it('should return an error when no collection match the identifier', () => { 101 | const server = new Database(); 102 | server.addCollection( 103 | 'foo', 104 | new Collection({ items: [{ id: 1, name: 'foo' }] }), 105 | ); 106 | expect(() => { 107 | server.getOne('foo', 2); 108 | }).toThrow(new Error('No item with identifier 2')); 109 | }); 110 | 111 | it('should return the first collection matching the identifier', () => { 112 | const server = new Database(); 113 | server.addCollection( 114 | 'foo', 115 | new Collection({ 116 | items: [ 117 | { id: 1, name: 'foo' }, 118 | { id: 2, name: 'bar' }, 119 | ], 120 | }), 121 | ); 122 | expect(server.getOne('foo', 1)).toEqual({ id: 1, name: 'foo' }); 123 | expect(server.getOne('foo', 2)).toEqual({ id: 2, name: 'bar' }); 124 | }); 125 | 126 | it('should use the identifierName', () => { 127 | const server = new Database(); 128 | server.addCollection( 129 | 'foo', 130 | new Collection({ 131 | items: [ 132 | { _id: 1, name: 'foo' }, 133 | { _id: 2, name: 'bar' }, 134 | ], 135 | identifierName: '_id', 136 | }), 137 | ); 138 | expect(server.getOne('foo', 1)).toEqual({ _id: 1, name: 'foo' }); 139 | expect(server.getOne('foo', 2)).toEqual({ _id: 2, name: 'bar' }); 140 | }); 141 | }); 142 | 143 | describe('getOnly', () => { 144 | it('should return the single matching the identifier', () => { 145 | const server = new Database(); 146 | server.addSingle('foo', new Single({ name: 'foo' })); 147 | expect(server.getOnly('foo')).toEqual({ name: 'foo' }); 148 | }); 149 | }); 150 | }); 151 | -------------------------------------------------------------------------------- /src/adapters/SinonAdapter.ts: -------------------------------------------------------------------------------- 1 | import type { SinonFakeXMLHttpRequest } from 'sinon'; 2 | import { 3 | SimpleRestServer, 4 | type BaseServerOptions, 5 | } from '../SimpleRestServer.ts'; 6 | import { parseQueryString } from '../parseQueryString.ts'; 7 | import type { BaseResponse, APIServer, NormalizedRequest } from '../types.ts'; 8 | 9 | export class SinonAdapter { 10 | loggingEnabled = false; 11 | server: APIServer; 12 | 13 | constructor({ 14 | loggingEnabled = false, 15 | server, 16 | ...options 17 | }: SinonAdapterOptions = {}) { 18 | this.server = server || new SimpleRestServer(options); 19 | this.loggingEnabled = loggingEnabled; 20 | } 21 | 22 | getHandler() { 23 | return async (request: SinonFakeXMLHttpRequest) => { 24 | // This is an internal property of SinonFakeXMLHttpRequest but we have to set it to 4 to 25 | // suppress sinon's synchronous processing (which would result in HTTP 404). This allows us 26 | // to handle the request asynchronously. 27 | // See https://github.com/sinonjs/sinon/issues/637 28 | // @ts-expect-error 29 | request.readyState = 4; 30 | const normalizedRequest = this.getNormalizedRequest(request); 31 | const response = await this.server.handle(normalizedRequest); 32 | this.respond(response, request); 33 | }; 34 | } 35 | 36 | getNormalizedRequest(request: SinonFakeXMLHttpRequest): NormalizedRequest { 37 | const req: Request | SinonFakeXMLHttpRequest = 38 | typeof request === 'string' ? new Request(request) : request; 39 | 40 | const queryString = req.url 41 | ? decodeURIComponent(req.url.slice(req.url.indexOf('?') + 1)) 42 | : ''; 43 | const params = parseQueryString(queryString); 44 | let requestBody: Record | undefined = undefined; 45 | if ((req as SinonFakeXMLHttpRequest).requestBody) { 46 | try { 47 | requestBody = JSON.parse( 48 | (req as SinonFakeXMLHttpRequest).requestBody, 49 | ); 50 | } catch (error) { 51 | // body isn't JSON, skipping 52 | } 53 | } 54 | 55 | return { 56 | url: req.url, 57 | headers: new Headers(request.requestHeaders), 58 | params, 59 | requestBody, 60 | method: req.method, 61 | }; 62 | } 63 | 64 | respond(response: BaseResponse, request: SinonFakeXMLHttpRequest) { 65 | const sinonResponse = { 66 | status: response.status, 67 | body: response.body ?? '', 68 | headers: response.headers ?? {}, 69 | }; 70 | 71 | if (Array.isArray(sinonResponse.headers)) { 72 | if ( 73 | !( 74 | sinonResponse.headers as Array<{ 75 | name: string; 76 | value: string; 77 | }> 78 | ).find((header) => header.name.toLowerCase() === 'content-type') 79 | ) { 80 | sinonResponse.headers.push({ 81 | name: 'Content-Type', 82 | value: 'application/json', 83 | }); 84 | } 85 | } else if ( 86 | !(sinonResponse.headers as Record)['Content-Type'] 87 | ) { 88 | sinonResponse.headers['Content-Type'] = 'application/json'; 89 | } 90 | 91 | // This is an internal property of SinonFakeXMLHttpRequest but we have to reset it to 1 92 | // to handle the request asynchronously. 93 | // See https://github.com/sinonjs/sinon/issues/637 94 | // @ts-expect-error 95 | request.readyState = 1; 96 | 97 | request.respond( 98 | sinonResponse.status, 99 | sinonResponse.headers, 100 | JSON.stringify(sinonResponse.body), 101 | ); 102 | 103 | this.log(request, sinonResponse); 104 | } 105 | 106 | log(request: SinonFakeXMLHttpRequest, response: SinonFakeRestResponse) { 107 | if (!this.loggingEnabled) return; 108 | if (console.group) { 109 | // Better logging in Chrome 110 | console.groupCollapsed(request.method, request.url, '(FakeRest)'); 111 | console.group('request'); 112 | console.log(request.method, request.url); 113 | console.log('headers', request.requestHeaders); 114 | console.log('body ', request.requestBody); 115 | console.groupEnd(); 116 | console.group('response', response.status); 117 | console.log('headers', response.headers); 118 | console.log('body ', response.body); 119 | console.groupEnd(); 120 | console.groupEnd(); 121 | } else { 122 | console.log( 123 | 'FakeRest request ', 124 | request.method, 125 | request.url, 126 | 'headers', 127 | request.requestHeaders, 128 | 'body', 129 | request.requestBody, 130 | ); 131 | console.log( 132 | 'FakeRest response', 133 | response.status, 134 | 'headers', 135 | response.headers, 136 | 'body', 137 | response.body, 138 | ); 139 | } 140 | } 141 | 142 | toggleLogging() { 143 | this.loggingEnabled = !this.loggingEnabled; 144 | } 145 | } 146 | 147 | export const getSinonHandler = (options: SinonAdapterOptions) => { 148 | const server = new SinonAdapter(options); 149 | return server.getHandler(); 150 | }; 151 | 152 | /** 153 | * @deprecated Use SinonServer instead 154 | */ 155 | export const Server = SinonAdapter; 156 | 157 | export type SinonFakeRestResponse = { 158 | status: number; 159 | body: any; 160 | headers: Record; 161 | }; 162 | 163 | export type SinonAdapterOptions = BaseServerOptions & { 164 | server?: APIServer; 165 | loggingEnabled?: boolean; 166 | }; 167 | -------------------------------------------------------------------------------- /src/Single.spec.ts: -------------------------------------------------------------------------------- 1 | import { Single } from './Single.ts'; 2 | import { Collection } from './Collection.ts'; 3 | import { Database } from './Database.ts'; 4 | 5 | describe('Single', () => { 6 | describe('constructor', () => { 7 | it('should set the intial set of data', () => { 8 | const single = new Single({ foo: 'bar' }); 9 | expect(single.getOnly()).toEqual({ foo: 'bar' }); 10 | }); 11 | }); 12 | 13 | describe('getOnly', () => { 14 | it('should return the passed in object', () => { 15 | const single = new Single({ foo: 'bar' }); 16 | expect(single.getOnly()).toEqual({ foo: 'bar' }); 17 | }); 18 | 19 | describe('embed query', () => { 20 | it('should throw an error when trying to embed a non-existing collection', () => { 21 | const foo = new Single({ name: 'foo', bar_id: 123 }); 22 | const database = new Database(); 23 | database.addSingle('foo', foo); 24 | expect(() => { 25 | foo.getOnly({ embed: ['bar'] }); 26 | }).toThrow( 27 | new Error("Can't embed a non-existing collection bar"), 28 | ); 29 | }); 30 | 31 | it('should return the original object for missing embed one', () => { 32 | const foo = new Single({ name: 'foo', bar_id: 123 }); 33 | const bars = new Collection({ items: [] }); 34 | const database = new Database(); 35 | database.addSingle('foo', foo); 36 | database.addCollection('bars', bars); 37 | const expected = { name: 'foo', bar_id: 123 }; 38 | expect(foo.getOnly({ embed: ['bar'] })).toEqual(expected); 39 | }); 40 | 41 | it('should return the object with the reference object for embed one', () => { 42 | const foo = new Single({ name: 'foo', bar_id: 123 }); 43 | const bars = new Collection({ 44 | items: [ 45 | { id: 1, bar: 'nobody wants me' }, 46 | { id: 123, bar: 'baz' }, 47 | { id: 456, bar: 'bazz' }, 48 | ], 49 | }); 50 | const database = new Database(); 51 | database.addSingle('foo', foo); 52 | database.addCollection('bars', bars); 53 | const expected = { 54 | name: 'foo', 55 | bar_id: 123, 56 | bar: { id: 123, bar: 'baz' }, 57 | }; 58 | expect(foo.getOnly({ embed: ['bar'] })).toEqual(expected); 59 | }); 60 | 61 | it('should throw an error when trying to embed many a non-existing collection', () => { 62 | const foo = new Single({ name: 'foo', bar_id: 123 }); 63 | const database = new Database(); 64 | database.addSingle('foo', foo); 65 | expect(() => { 66 | foo.getOnly({ embed: ['bars'] }); 67 | }).toThrow( 68 | new Error("Can't embed a non-existing collection bars"), 69 | ); 70 | }); 71 | 72 | it('should return the object with an array of references for embed many using inner array', () => { 73 | const foo = new Single({ name: 'foo', bars: [1, 3] }); 74 | const bars = new Collection({ 75 | items: [ 76 | { id: 1, bar: 'baz' }, 77 | { id: 2, bar: 'biz' }, 78 | { id: 3, bar: 'boz' }, 79 | ], 80 | }); 81 | const database = new Database(); 82 | database.addSingle('foo', foo); 83 | database.addCollection('bars', bars); 84 | const expected = { 85 | name: 'foo', 86 | bars: [ 87 | { id: 1, bar: 'baz' }, 88 | { id: 3, bar: 'boz' }, 89 | ], 90 | }; 91 | expect(foo.getOnly({ embed: ['bars'] })).toEqual(expected); 92 | }); 93 | 94 | it('should allow multiple embeds', () => { 95 | const foo = new Single({ 96 | name: 'foo', 97 | bars: [1, 3], 98 | bazs: [4, 5], 99 | }); 100 | const bars = new Collection({ 101 | items: [ 102 | { id: 1, name: 'bar1' }, 103 | { id: 2, name: 'bar2' }, 104 | { id: 3, name: 'bar3' }, 105 | ], 106 | }); 107 | const bazs = new Collection({ 108 | items: [ 109 | { id: 4, name: 'baz1' }, 110 | { id: 5, name: 'baz2' }, 111 | { id: 6, name: 'baz3' }, 112 | ], 113 | }); 114 | const database = new Database(); 115 | database.addSingle('foo', foo); 116 | database.addCollection('bars', bars); 117 | database.addCollection('bazs', bazs); 118 | const expected = { 119 | name: 'foo', 120 | bars: [ 121 | { id: 1, name: 'bar1' }, 122 | { id: 3, name: 'bar3' }, 123 | ], 124 | bazs: [ 125 | { id: 4, name: 'baz1' }, 126 | { id: 5, name: 'baz2' }, 127 | ], 128 | }; 129 | expect(foo.getOnly({ embed: ['bars', 'bazs'] })).toEqual( 130 | expected, 131 | ); 132 | }); 133 | }); 134 | }); 135 | 136 | describe('updateOnly', () => { 137 | it('should return the updated item', () => { 138 | const single = new Single({ name: 'foo' }); 139 | expect(single.updateOnly({ name: 'bar' })).toEqual({ name: 'bar' }); 140 | }); 141 | 142 | it('should update the item', () => { 143 | const single = new Single({ name: 'foo' }); 144 | single.updateOnly({ name: 'bar' }); 145 | expect(single.getOnly()).toEqual({ name: 'bar' }); 146 | }); 147 | 148 | it('should not update the original item', () => { 149 | const data = { name: 'foo' }; 150 | const single = new Single(data); 151 | single.updateOnly({ name: 'bar' }); 152 | expect(single.getOnly()).toEqual({ name: 'bar' }); 153 | expect(data).toEqual({ name: 'foo' }); 154 | }); 155 | }); 156 | }); 157 | -------------------------------------------------------------------------------- /src/SimpleRestServer.spec.ts: -------------------------------------------------------------------------------- 1 | import { SimpleRestServer } from './SimpleRestServer.ts'; 2 | 3 | describe('SimpleRestServer', () => { 4 | describe('getAll', () => { 5 | it('should return list results according to the request parameters', async () => { 6 | const data = { 7 | posts: [ 8 | { 9 | id: 1, 10 | title: 'bazingaaa', 11 | }, 12 | { 13 | id: 2, 14 | title: 'bazinga', 15 | }, 16 | { 17 | id: 3, 18 | title: 'nope', 19 | }, 20 | ], 21 | }; 22 | 23 | const server = new SimpleRestServer({ 24 | baseUrl: 'http://localhost:4000', 25 | data, 26 | }); 27 | 28 | const response = await server.handleRequest({ 29 | url: 'http://localhost:4000/posts', 30 | method: 'GET', 31 | params: { 32 | filter: { q: 'bazin' }, 33 | range: [0, 1], 34 | sort: 'title', 35 | }, 36 | requestBody: undefined, 37 | }); 38 | 39 | expect(response).toEqual( 40 | expect.objectContaining({ 41 | status: 200, 42 | body: [ 43 | { 44 | id: 2, 45 | title: 'bazinga', 46 | }, 47 | { 48 | id: 1, 49 | title: 'bazingaaa', 50 | }, 51 | ], 52 | }), 53 | ); 54 | }); 55 | }); 56 | describe('getOne', () => { 57 | it('should correctly get records with a numeric identifier', async () => { 58 | const data = { 59 | posts: [ 60 | { 61 | id: 1, 62 | title: 'test', 63 | }, 64 | ], 65 | }; 66 | 67 | const server = new SimpleRestServer({ 68 | baseUrl: 'http://localhost:4000', 69 | data, 70 | }); 71 | 72 | const response = await server.handleRequest({ 73 | url: 'http://localhost:4000/posts/1', 74 | method: 'GET', 75 | params: {}, 76 | requestBody: undefined, 77 | }); 78 | 79 | expect(response).toEqual( 80 | expect.objectContaining({ 81 | status: 200, 82 | body: { 83 | id: 1, 84 | title: 'test', 85 | }, 86 | }), 87 | ); 88 | }); 89 | it('should correctly get records with a string identifier', async () => { 90 | const data = { 91 | posts: [ 92 | { 93 | id: 'bazinga', 94 | title: 'test', 95 | }, 96 | ], 97 | }; 98 | 99 | const server = new SimpleRestServer({ 100 | baseUrl: 'http://localhost:4000', 101 | data, 102 | }); 103 | 104 | const response = await server.handleRequest({ 105 | url: 'http://localhost:4000/posts/bazinga', 106 | method: 'GET', 107 | params: {}, 108 | requestBody: undefined, 109 | }); 110 | 111 | expect(response).toEqual( 112 | expect.objectContaining({ 113 | status: 200, 114 | body: { 115 | id: 'bazinga', 116 | title: 'test', 117 | }, 118 | }), 119 | ); 120 | }); 121 | }); 122 | describe('update', () => { 123 | it('should correctly update records with a numeric identifier', async () => { 124 | const data = { 125 | posts: [ 126 | { 127 | id: 1, 128 | title: 'test', 129 | }, 130 | ], 131 | }; 132 | 133 | const server = new SimpleRestServer({ 134 | baseUrl: 'http://localhost:4000', 135 | data, 136 | }); 137 | 138 | const response = await server.handleRequest({ 139 | url: 'http://localhost:4000/posts/1', 140 | method: 'PUT', 141 | params: {}, 142 | requestBody: { 143 | id: 1, 144 | title: 'test42', 145 | }, 146 | }); 147 | 148 | expect(response).toEqual( 149 | expect.objectContaining({ 150 | status: 200, 151 | body: { 152 | id: 1, 153 | title: 'test42', 154 | }, 155 | }), 156 | ); 157 | }); 158 | it('should correctly update records with a string identifier', async () => { 159 | const data = { 160 | posts: [ 161 | { 162 | id: 'bazinga', 163 | title: 'test', 164 | }, 165 | ], 166 | }; 167 | 168 | const server = new SimpleRestServer({ 169 | baseUrl: 'http://localhost:4000', 170 | data, 171 | }); 172 | 173 | const response = await server.handleRequest({ 174 | url: 'http://localhost:4000/posts/bazinga', 175 | method: 'PUT', 176 | params: {}, 177 | requestBody: { 178 | id: 'bazinga', 179 | title: 'test42', 180 | }, 181 | }); 182 | 183 | expect(response).toEqual( 184 | expect.objectContaining({ 185 | status: 200, 186 | body: { 187 | id: 'bazinga', 188 | title: 'test42', 189 | }, 190 | }), 191 | ); 192 | }); 193 | }); 194 | describe('delete', () => { 195 | it('should correctly delete records with a numeric identifier', async () => { 196 | const data = { 197 | posts: [ 198 | { 199 | id: 1, 200 | title: 'test', 201 | }, 202 | ], 203 | }; 204 | 205 | const server = new SimpleRestServer({ 206 | baseUrl: 'http://localhost:4000', 207 | data, 208 | }); 209 | 210 | const response = await server.handleRequest({ 211 | url: 'http://localhost:4000/posts/1', 212 | method: 'DELETE', 213 | params: {}, 214 | requestBody: undefined, 215 | }); 216 | 217 | expect(response).toEqual( 218 | expect.objectContaining({ 219 | status: 200, 220 | body: { 221 | id: 1, 222 | title: 'test', 223 | }, 224 | }), 225 | ); 226 | }); 227 | it('should correctly delete records with a string identifier', async () => { 228 | const data = { 229 | posts: [ 230 | { 231 | id: 'bazinga', 232 | title: 'test', 233 | }, 234 | ], 235 | }; 236 | 237 | const server = new SimpleRestServer({ 238 | baseUrl: 'http://localhost:4000', 239 | data, 240 | }); 241 | 242 | const response = await server.handleRequest({ 243 | url: 'http://localhost:4000/posts/bazinga', 244 | method: 'DELETE', 245 | params: {}, 246 | requestBody: { 247 | id: 'bazinga', 248 | title: 'test', 249 | }, 250 | }); 251 | 252 | expect(response).toEqual( 253 | expect.objectContaining({ 254 | status: 200, 255 | body: { 256 | id: 'bazinga', 257 | title: 'test', 258 | }, 259 | }), 260 | ); 261 | }); 262 | }); 263 | }); 264 | -------------------------------------------------------------------------------- /public/mockServiceWorker.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | /* tslint:disable */ 3 | 4 | /** 5 | * Mock Service Worker. 6 | * @see https://github.com/mswjs/msw 7 | * - Please do NOT modify this file. 8 | */ 9 | 10 | const PACKAGE_VERSION = '2.10.5' 11 | const INTEGRITY_CHECKSUM = 'f5825c521429caf22a4dd13b66e243af' 12 | const IS_MOCKED_RESPONSE = Symbol('isMockedResponse') 13 | const activeClientIds = new Set() 14 | 15 | addEventListener('install', function () { 16 | self.skipWaiting() 17 | }) 18 | 19 | addEventListener('activate', function (event) { 20 | event.waitUntil(self.clients.claim()) 21 | }) 22 | 23 | addEventListener('message', async function (event) { 24 | const clientId = Reflect.get(event.source || {}, 'id') 25 | 26 | if (!clientId || !self.clients) { 27 | return 28 | } 29 | 30 | const client = await self.clients.get(clientId) 31 | 32 | if (!client) { 33 | return 34 | } 35 | 36 | const allClients = await self.clients.matchAll({ 37 | type: 'window', 38 | }) 39 | 40 | switch (event.data) { 41 | case 'KEEPALIVE_REQUEST': { 42 | sendToClient(client, { 43 | type: 'KEEPALIVE_RESPONSE', 44 | }) 45 | break 46 | } 47 | 48 | case 'INTEGRITY_CHECK_REQUEST': { 49 | sendToClient(client, { 50 | type: 'INTEGRITY_CHECK_RESPONSE', 51 | payload: { 52 | packageVersion: PACKAGE_VERSION, 53 | checksum: INTEGRITY_CHECKSUM, 54 | }, 55 | }) 56 | break 57 | } 58 | 59 | case 'MOCK_ACTIVATE': { 60 | activeClientIds.add(clientId) 61 | 62 | sendToClient(client, { 63 | type: 'MOCKING_ENABLED', 64 | payload: { 65 | client: { 66 | id: client.id, 67 | frameType: client.frameType, 68 | }, 69 | }, 70 | }) 71 | break 72 | } 73 | 74 | case 'MOCK_DEACTIVATE': { 75 | activeClientIds.delete(clientId) 76 | break 77 | } 78 | 79 | case 'CLIENT_CLOSED': { 80 | activeClientIds.delete(clientId) 81 | 82 | const remainingClients = allClients.filter((client) => { 83 | return client.id !== clientId 84 | }) 85 | 86 | // Unregister itself when there are no more clients 87 | if (remainingClients.length === 0) { 88 | self.registration.unregister() 89 | } 90 | 91 | break 92 | } 93 | } 94 | }) 95 | 96 | addEventListener('fetch', function (event) { 97 | // Bypass navigation requests. 98 | if (event.request.mode === 'navigate') { 99 | return 100 | } 101 | 102 | // Opening the DevTools triggers the "only-if-cached" request 103 | // that cannot be handled by the worker. Bypass such requests. 104 | if ( 105 | event.request.cache === 'only-if-cached' && 106 | event.request.mode !== 'same-origin' 107 | ) { 108 | return 109 | } 110 | 111 | // Bypass all requests when there are no active clients. 112 | // Prevents the self-unregistered worked from handling requests 113 | // after it's been deleted (still remains active until the next reload). 114 | if (activeClientIds.size === 0) { 115 | return 116 | } 117 | 118 | const requestId = crypto.randomUUID() 119 | event.respondWith(handleRequest(event, requestId)) 120 | }) 121 | 122 | /** 123 | * @param {FetchEvent} event 124 | * @param {string} requestId 125 | */ 126 | async function handleRequest(event, requestId) { 127 | const client = await resolveMainClient(event) 128 | const requestCloneForEvents = event.request.clone() 129 | const response = await getResponse(event, client, requestId) 130 | 131 | // Send back the response clone for the "response:*" life-cycle events. 132 | // Ensure MSW is active and ready to handle the message, otherwise 133 | // this message will pend indefinitely. 134 | if (client && activeClientIds.has(client.id)) { 135 | const serializedRequest = await serializeRequest(requestCloneForEvents) 136 | 137 | // Clone the response so both the client and the library could consume it. 138 | const responseClone = response.clone() 139 | 140 | sendToClient( 141 | client, 142 | { 143 | type: 'RESPONSE', 144 | payload: { 145 | isMockedResponse: IS_MOCKED_RESPONSE in response, 146 | request: { 147 | id: requestId, 148 | ...serializedRequest, 149 | }, 150 | response: { 151 | type: responseClone.type, 152 | status: responseClone.status, 153 | statusText: responseClone.statusText, 154 | headers: Object.fromEntries(responseClone.headers.entries()), 155 | body: responseClone.body, 156 | }, 157 | }, 158 | }, 159 | responseClone.body ? [serializedRequest.body, responseClone.body] : [], 160 | ) 161 | } 162 | 163 | return response 164 | } 165 | 166 | /** 167 | * Resolve the main client for the given event. 168 | * Client that issues a request doesn't necessarily equal the client 169 | * that registered the worker. It's with the latter the worker should 170 | * communicate with during the response resolving phase. 171 | * @param {FetchEvent} event 172 | * @returns {Promise} 173 | */ 174 | async function resolveMainClient(event) { 175 | const client = await self.clients.get(event.clientId) 176 | 177 | if (activeClientIds.has(event.clientId)) { 178 | return client 179 | } 180 | 181 | if (client?.frameType === 'top-level') { 182 | return client 183 | } 184 | 185 | const allClients = await self.clients.matchAll({ 186 | type: 'window', 187 | }) 188 | 189 | return allClients 190 | .filter((client) => { 191 | // Get only those clients that are currently visible. 192 | return client.visibilityState === 'visible' 193 | }) 194 | .find((client) => { 195 | // Find the client ID that's recorded in the 196 | // set of clients that have registered the worker. 197 | return activeClientIds.has(client.id) 198 | }) 199 | } 200 | 201 | /** 202 | * @param {FetchEvent} event 203 | * @param {Client | undefined} client 204 | * @param {string} requestId 205 | * @returns {Promise} 206 | */ 207 | async function getResponse(event, client, requestId) { 208 | // Clone the request because it might've been already used 209 | // (i.e. its body has been read and sent to the client). 210 | const requestClone = event.request.clone() 211 | 212 | function passthrough() { 213 | // Cast the request headers to a new Headers instance 214 | // so the headers can be manipulated with. 215 | const headers = new Headers(requestClone.headers) 216 | 217 | // Remove the "accept" header value that marked this request as passthrough. 218 | // This prevents request alteration and also keeps it compliant with the 219 | // user-defined CORS policies. 220 | const acceptHeader = headers.get('accept') 221 | if (acceptHeader) { 222 | const values = acceptHeader.split(',').map((value) => value.trim()) 223 | const filteredValues = values.filter( 224 | (value) => value !== 'msw/passthrough', 225 | ) 226 | 227 | if (filteredValues.length > 0) { 228 | headers.set('accept', filteredValues.join(', ')) 229 | } else { 230 | headers.delete('accept') 231 | } 232 | } 233 | 234 | return fetch(requestClone, { headers }) 235 | } 236 | 237 | // Bypass mocking when the client is not active. 238 | if (!client) { 239 | return passthrough() 240 | } 241 | 242 | // Bypass initial page load requests (i.e. static assets). 243 | // The absence of the immediate/parent client in the map of the active clients 244 | // means that MSW hasn't dispatched the "MOCK_ACTIVATE" event yet 245 | // and is not ready to handle requests. 246 | if (!activeClientIds.has(client.id)) { 247 | return passthrough() 248 | } 249 | 250 | // Notify the client that a request has been intercepted. 251 | const serializedRequest = await serializeRequest(event.request) 252 | const clientMessage = await sendToClient( 253 | client, 254 | { 255 | type: 'REQUEST', 256 | payload: { 257 | id: requestId, 258 | ...serializedRequest, 259 | }, 260 | }, 261 | [serializedRequest.body], 262 | ) 263 | 264 | switch (clientMessage.type) { 265 | case 'MOCK_RESPONSE': { 266 | return respondWithMock(clientMessage.data) 267 | } 268 | 269 | case 'PASSTHROUGH': { 270 | return passthrough() 271 | } 272 | } 273 | 274 | return passthrough() 275 | } 276 | 277 | /** 278 | * @param {Client} client 279 | * @param {any} message 280 | * @param {Array} transferrables 281 | * @returns {Promise} 282 | */ 283 | function sendToClient(client, message, transferrables = []) { 284 | return new Promise((resolve, reject) => { 285 | const channel = new MessageChannel() 286 | 287 | channel.port1.onmessage = (event) => { 288 | if (event.data && event.data.error) { 289 | return reject(event.data.error) 290 | } 291 | 292 | resolve(event.data) 293 | } 294 | 295 | client.postMessage(message, [ 296 | channel.port2, 297 | ...transferrables.filter(Boolean), 298 | ]) 299 | }) 300 | } 301 | 302 | /** 303 | * @param {Response} response 304 | * @returns {Response} 305 | */ 306 | function respondWithMock(response) { 307 | // Setting response status code to 0 is a no-op. 308 | // However, when responding with a "Response.error()", the produced Response 309 | // instance will have status code set to 0. Since it's not possible to create 310 | // a Response instance with status code 0, handle that use-case separately. 311 | if (response.status === 0) { 312 | return Response.error() 313 | } 314 | 315 | const mockedResponse = new Response(response.body, response) 316 | 317 | Reflect.defineProperty(mockedResponse, IS_MOCKED_RESPONSE, { 318 | value: true, 319 | enumerable: true, 320 | }) 321 | 322 | return mockedResponse 323 | } 324 | 325 | /** 326 | * @param {Request} request 327 | */ 328 | async function serializeRequest(request) { 329 | return { 330 | url: request.url, 331 | mode: request.mode, 332 | method: request.method, 333 | headers: Object.fromEntries(request.headers.entries()), 334 | cache: request.cache, 335 | credentials: request.credentials, 336 | destination: request.destination, 337 | integrity: request.integrity, 338 | redirect: request.redirect, 339 | referrer: request.referrer, 340 | referrerPolicy: request.referrerPolicy, 341 | body: await request.arrayBuffer(), 342 | keepalive: request.keepalive, 343 | } 344 | } 345 | -------------------------------------------------------------------------------- /src/SimpleRestServer.ts: -------------------------------------------------------------------------------- 1 | import type { Collection } from './Collection.ts'; 2 | import { Database, type DatabaseOptions } from './Database.ts'; 3 | import type { Single } from './Single.ts'; 4 | import type { 5 | APIServer, 6 | BaseResponse, 7 | FakeRestContext, 8 | CollectionItem, 9 | QueryFunction, 10 | NormalizedRequest, 11 | } from './types.ts'; 12 | 13 | export class SimpleRestServer implements APIServer { 14 | baseUrl = ''; 15 | defaultQuery: QueryFunction = () => ({}); 16 | middlewares: Array; 17 | database: Database; 18 | 19 | constructor({ 20 | baseUrl = '', 21 | defaultQuery = () => ({}), 22 | database, 23 | middlewares, 24 | ...options 25 | }: BaseServerOptions = {}) { 26 | this.baseUrl = baseUrl; 27 | this.defaultQuery = defaultQuery; 28 | this.middlewares = middlewares || []; 29 | 30 | if (database) { 31 | this.database = database; 32 | } else { 33 | this.database = new Database(options); 34 | } 35 | } 36 | 37 | /** 38 | * @param Function ResourceName => object 39 | */ 40 | setDefaultQuery(query: QueryFunction) { 41 | this.defaultQuery = query; 42 | } 43 | 44 | getContext(normalizedRequest: NormalizedRequest): FakeRestContext { 45 | for (const name of this.database.getSingleNames()) { 46 | const matches = normalizedRequest.url?.match( 47 | new RegExp(`^${this.baseUrl}\\/(${name})(\\/?.*)?$`), 48 | ); 49 | if (!matches) continue; 50 | return { 51 | ...normalizedRequest, 52 | single: name, 53 | }; 54 | } 55 | 56 | const matches = normalizedRequest.url?.match( 57 | new RegExp( 58 | `^${this.baseUrl}\\/([^\\/?]+)(\\/(\\w+|\\d+))?(\\?.*)?$`, 59 | ), 60 | ); 61 | if (matches) { 62 | const name = matches[1]; 63 | const params = Object.assign( 64 | {}, 65 | this.defaultQuery(name), 66 | normalizedRequest.params, 67 | ); 68 | 69 | return { 70 | ...normalizedRequest, 71 | collection: name, 72 | params, 73 | }; 74 | } 75 | 76 | return normalizedRequest; 77 | } 78 | 79 | async handle(normalizedRequest: NormalizedRequest): Promise { 80 | const context = this.getContext(normalizedRequest); 81 | // Call middlewares 82 | let index = 0; 83 | const middlewares = [...this.middlewares]; 84 | 85 | const next = (context: FakeRestContext) => { 86 | const middleware = middlewares[index++]; 87 | if (middleware) { 88 | return middleware(context, next); 89 | } 90 | return this.handleRequest(context); 91 | }; 92 | 93 | try { 94 | const response = await next(context); 95 | return response; 96 | } catch (error) { 97 | if (error instanceof Error) { 98 | throw error; 99 | } 100 | 101 | return error as BaseResponse; 102 | } 103 | } 104 | 105 | handleRequest(context: FakeRestContext): BaseResponse { 106 | // Handle Single Objects 107 | for (const name of this.database.getSingleNames()) { 108 | const matches = context.url?.match( 109 | new RegExp(`^${this.baseUrl}\\/(${name})(\\/?.*)?$`), 110 | ); 111 | if (!matches) continue; 112 | 113 | if (context.method === 'GET') { 114 | try { 115 | return { 116 | status: 200, 117 | body: this.database.getOnly(name), 118 | headers: { 119 | 'Content-Type': 'application/json', 120 | }, 121 | }; 122 | } catch (error) { 123 | return { 124 | status: 404, 125 | headers: {}, 126 | }; 127 | } 128 | } 129 | if (context.method === 'PUT') { 130 | try { 131 | if (context.requestBody == null) { 132 | return { 133 | status: 400, 134 | headers: {}, 135 | }; 136 | } 137 | return { 138 | status: 200, 139 | body: this.database.updateOnly( 140 | name, 141 | context.requestBody, 142 | ), 143 | headers: { 144 | 'Content-Type': 'application/json', 145 | }, 146 | }; 147 | } catch (error) { 148 | return { 149 | status: 404, 150 | headers: {}, 151 | }; 152 | } 153 | } 154 | if (context.method === 'PATCH') { 155 | try { 156 | if (context.requestBody == null) { 157 | return { 158 | status: 400, 159 | headers: {}, 160 | }; 161 | } 162 | return { 163 | status: 200, 164 | body: this.database.updateOnly( 165 | name, 166 | context.requestBody, 167 | ), 168 | headers: { 169 | 'Content-Type': 'application/json', 170 | }, 171 | }; 172 | } catch (error) { 173 | return { 174 | status: 404, 175 | headers: {}, 176 | }; 177 | } 178 | } 179 | } 180 | 181 | // handle collections 182 | const matches = context.url?.match( 183 | new RegExp( 184 | `^${this.baseUrl}\\/([^\\/?]+)(\\/(\\w+|\\d+))?(\\?.*)?$`, 185 | ), 186 | ); 187 | if (!matches) { 188 | return { status: 404, headers: {} }; 189 | } 190 | const name = matches[1]; 191 | const params = Object.assign( 192 | {}, 193 | this.defaultQuery(name), 194 | context.params, 195 | ); 196 | if (!matches[2]) { 197 | if (context.method === 'GET') { 198 | if (!this.database.getCollection(name)) { 199 | return { status: 404, headers: {} }; 200 | } 201 | const count = this.database.getCount( 202 | name, 203 | params.filter || params.embed 204 | ? { filter: params.filter, embed: params.embed } 205 | : {}, 206 | ); 207 | if (count > 0) { 208 | const items = this.database.getAll(name, params); 209 | const first = params.range ? params.range[0] : 0; 210 | const last = 211 | params.range && params.range.length === 2 212 | ? Math.min( 213 | items.length - 1 + first, 214 | params.range[1], 215 | ) 216 | : items.length - 1; 217 | 218 | return { 219 | status: items.length === count ? 200 : 206, 220 | body: items, 221 | headers: { 222 | 'Content-Type': 'application/json', 223 | 'Content-Range': `items ${first}-${last}/${count}`, 224 | }, 225 | }; 226 | } 227 | 228 | return { 229 | status: 200, 230 | body: [], 231 | headers: { 232 | 'Content-Type': 'application/json', 233 | 'Content-Range': 'items */0', 234 | }, 235 | }; 236 | } 237 | if (context.method === 'POST') { 238 | if (context.requestBody == null) { 239 | return { 240 | status: 400, 241 | headers: {}, 242 | }; 243 | } 244 | 245 | const newResource = this.database.addOne( 246 | name, 247 | context.requestBody, 248 | ); 249 | const newResourceURI = `${this.baseUrl}/${name}/${ 250 | newResource[ 251 | this.database.getCollection(name).identifierName 252 | ] 253 | }`; 254 | 255 | return { 256 | status: 201, 257 | body: newResource, 258 | headers: { 259 | 'Content-Type': 'application/json', 260 | Location: newResourceURI, 261 | }, 262 | }; 263 | } 264 | } else { 265 | if (!this.database.getCollection(name)) { 266 | return { status: 404, headers: {} }; 267 | } 268 | const id = matches[3]; 269 | if (context.method === 'GET') { 270 | try { 271 | return { 272 | status: 200, 273 | body: this.database.getOne(name, id, params), 274 | headers: { 275 | 'Content-Type': 'application/json', 276 | }, 277 | }; 278 | } catch (error) { 279 | return { 280 | status: 404, 281 | headers: {}, 282 | }; 283 | } 284 | } 285 | if (context.method === 'PUT') { 286 | try { 287 | if (context.requestBody == null) { 288 | return { 289 | status: 400, 290 | headers: {}, 291 | }; 292 | } 293 | return { 294 | status: 200, 295 | body: this.database.updateOne( 296 | name, 297 | id, 298 | context.requestBody, 299 | ), 300 | headers: { 301 | 'Content-Type': 'application/json', 302 | }, 303 | }; 304 | } catch (error) { 305 | return { 306 | status: 404, 307 | headers: {}, 308 | }; 309 | } 310 | } 311 | if (context.method === 'PATCH') { 312 | try { 313 | if (context.requestBody == null) { 314 | return { 315 | status: 400, 316 | headers: {}, 317 | }; 318 | } 319 | return { 320 | status: 200, 321 | body: this.database.updateOne( 322 | name, 323 | id, 324 | context.requestBody, 325 | ), 326 | headers: { 327 | 'Content-Type': 'application/json', 328 | }, 329 | }; 330 | } catch (error) { 331 | return { 332 | status: 404, 333 | headers: {}, 334 | }; 335 | } 336 | } 337 | if (context.method === 'DELETE') { 338 | try { 339 | return { 340 | status: 200, 341 | body: this.database.removeOne(name, id), 342 | headers: { 343 | 'Content-Type': 'application/json', 344 | }, 345 | }; 346 | } catch (error) { 347 | return { 348 | status: 404, 349 | headers: {}, 350 | }; 351 | } 352 | } 353 | } 354 | return { 355 | status: 404, 356 | headers: {}, 357 | }; 358 | } 359 | 360 | addMiddleware(middleware: Middleware) { 361 | this.middlewares.push(middleware); 362 | } 363 | 364 | addCollection( 365 | name: string, 366 | collection: Collection, 367 | ) { 368 | this.database.addCollection(name, collection); 369 | } 370 | 371 | getCollection(name: string) { 372 | return this.database.getCollection(name); 373 | } 374 | 375 | getCollectionNames() { 376 | return this.database.getCollectionNames(); 377 | } 378 | 379 | addSingle( 380 | name: string, 381 | single: Single, 382 | ) { 383 | this.database.addSingle(name, single); 384 | } 385 | 386 | getSingle(name: string) { 387 | return this.database.getSingle(name); 388 | } 389 | 390 | getSingleNames() { 391 | return this.database.getSingleNames(); 392 | } 393 | } 394 | 395 | export type Middleware = ( 396 | context: FakeRestContext, 397 | next: (context: FakeRestContext) => Promise | BaseResponse, 398 | ) => Promise | BaseResponse; 399 | 400 | export type BaseServerOptions = DatabaseOptions & { 401 | database?: Database; 402 | baseUrl?: string; 403 | batchUrl?: string | null; 404 | defaultQuery?: QueryFunction; 405 | middlewares?: Array; 406 | }; 407 | -------------------------------------------------------------------------------- /example/users.json: -------------------------------------------------------------------------------- 1 | { 2 | "users": [ 3 | { 4 | "id": 1, 5 | "username": "janedoe", 6 | "password": "password", 7 | "fullName": "Jane Doe", 8 | "avatar": "" 9 | }, 10 | { 11 | "id": 2, 12 | "username": "johndoe", 13 | "password": "password", 14 | "fullName": "John Doe", 15 | "avatar": "" 16 | } 17 | ] 18 | } 19 | -------------------------------------------------------------------------------- /src/Collection.ts: -------------------------------------------------------------------------------- 1 | import get from 'lodash/get.js'; 2 | import matches from 'lodash/matches.js'; 3 | import cloneDeep from 'lodash/cloneDeep.js'; 4 | import type { Database } from './Database.ts'; 5 | import type { 6 | CollectionItem, 7 | Embed, 8 | Filter, 9 | Predicate, 10 | Query, 11 | Range, 12 | Sort, 13 | } from './types.js'; 14 | 15 | export class Collection { 16 | sequence = 0; 17 | items: T[] = []; 18 | database: Database | null = null; 19 | name: string | null = null; 20 | identifierName = 'id'; 21 | getNewId: () => number | string; 22 | 23 | constructor({ 24 | items = [], 25 | identifierName = 'id', 26 | getNewId, 27 | }: { 28 | items?: T[]; 29 | identifierName?: string; 30 | getNewId?: () => number | string; 31 | } = {}) { 32 | if (!Array.isArray(items)) { 33 | throw new Error( 34 | "Can't initialize a Collection with anything else than an array of items", 35 | ); 36 | } 37 | this.identifierName = identifierName; 38 | this.getNewId = getNewId || this.getNewIdFromSequence; 39 | items.map(this.addOne.bind(this)); 40 | } 41 | 42 | /** 43 | * A Collection may need to access other collections (e.g. for embedding references) 44 | * This is done through a reference to the parent database. 45 | */ 46 | setDatabase(database: Database) { 47 | this.database = database; 48 | } 49 | 50 | setName(name: string) { 51 | this.name = name; 52 | } 53 | 54 | /** 55 | * Get a one to many embedder function for a given resource name 56 | * 57 | * @example embed posts for an author 58 | * 59 | * authorsCollection._oneToManyEmbedder('posts') 60 | * 61 | * @returns Function item => item 62 | */ 63 | _oneToManyEmbedder(resourceName: string) { 64 | if (this.name == null) { 65 | throw new Error("Can't embed references without a collection name"); 66 | } 67 | const singularResourceName = this.name.slice(0, -1); 68 | const referenceName = `${singularResourceName}_id`; 69 | return (item: T) => { 70 | if (this.database == null) { 71 | throw new Error("Can't embed references without a database"); 72 | } 73 | const otherCollection = this.database.collections[resourceName]; 74 | if (!otherCollection) 75 | throw new Error( 76 | `Can't embed a non-existing collection ${resourceName}`, 77 | ); 78 | if (Array.isArray(item[resourceName])) { 79 | // the many to one relationship is carried by an array of ids, e.g. { posts: [1, 2] } in authors 80 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 81 | item[resourceName] = otherCollection.getAll({ 82 | filter: (i: T) => 83 | item[resourceName].indexOf( 84 | i[otherCollection.identifierName], 85 | ) !== -1, 86 | }); 87 | } else { 88 | // the many to one relationship is carried by references in the related collection, e.g. { author_id: 1 } in posts 89 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 90 | item[resourceName] = otherCollection.getAll({ 91 | filter: (i: T) => 92 | i[referenceName] === item[this.identifierName], 93 | }); 94 | } 95 | return item; 96 | }; 97 | } 98 | 99 | /** 100 | * Get a many to one embedder function for a given resource name 101 | * 102 | * @example embed author for a post 103 | * 104 | * postsCollection._manyToOneEmbedder('author') 105 | * 106 | * @returns Function item => item 107 | */ 108 | _manyToOneEmbedder(resourceName: string) { 109 | const pluralResourceName = `${resourceName}s`; 110 | const referenceName = `${resourceName}_id`; 111 | return (item: T) => { 112 | if (this.database == null) { 113 | throw new Error("Can't embed references without a database"); 114 | } 115 | const otherCollection = 116 | this.database.collections[pluralResourceName]; 117 | if (!otherCollection) 118 | throw new Error( 119 | `Can't embed a non-existing collection ${resourceName}`, 120 | ); 121 | try { 122 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 123 | item[resourceName] = otherCollection.getOne( 124 | item[referenceName], 125 | ); 126 | } catch (e) { 127 | // resource doesn't exist in the related collection - do not embed 128 | } 129 | return item; 130 | }; 131 | } 132 | 133 | /** 134 | * @param String[] An array of resource names, e.g. ['books', 'country'] 135 | * @returns Function item => item 136 | */ 137 | _itemEmbedder(embed: Embed) { 138 | const resourceNames = Array.isArray(embed) ? embed : [embed]; 139 | const resourceEmbedders = resourceNames.map((resourceName) => 140 | resourceName.endsWith('s') 141 | ? this._oneToManyEmbedder(resourceName) 142 | : this._manyToOneEmbedder(resourceName), 143 | ); 144 | return (item: T) => 145 | resourceEmbedders.reduce( 146 | (itemWithEmbeds, embedder) => embedder(itemWithEmbeds), 147 | item, 148 | ); 149 | } 150 | 151 | getCount(query?: Query) { 152 | return this.getAll(query).length; 153 | } 154 | 155 | getAll(query?: Query) { 156 | let items = this.items.slice(0); // clone the array to avoid updating the core one 157 | if (query) { 158 | items = items.map((item) => Object.assign({}, item)); // clone item to avoid returning the original 159 | 160 | // Embed relationships first if requested 161 | if (query.embed && this.database) { 162 | items = items.map(this._itemEmbedder(query.embed)); 163 | } 164 | 165 | // Apply filter 166 | if (query.filter) { 167 | items = filterItems(items, query.filter); 168 | } 169 | 170 | // Apply sort 171 | if (query.sort) { 172 | items = sortItems(items, query.sort); 173 | } 174 | 175 | // Apply range 176 | if (query.range) { 177 | items = rangeItems(items, query.range); 178 | } 179 | } 180 | return items; 181 | } 182 | 183 | getIndex(identifier: number | string) { 184 | return this.items.findIndex( 185 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 186 | (item) => item[this.identifierName] == identifier, 187 | ); 188 | } 189 | 190 | getOne(identifier: number | string, query?: Query) { 191 | const index = this.getIndex(identifier); 192 | if (index === -1) { 193 | throw new Error(`No item with identifier ${identifier}`); 194 | } 195 | let item = this.items[index]; 196 | item = Object.assign({}, item); // clone item to avoid returning the original 197 | if (query?.embed && this.database) { 198 | item = this._itemEmbedder(query.embed)(item); // embed reference 199 | } 200 | return item; 201 | } 202 | 203 | getNewIdFromSequence() { 204 | return this.sequence++; 205 | } 206 | 207 | addOne(item: T) { 208 | const clone = cloneDeep(item); 209 | const identifier = clone[this.identifierName]; 210 | if (identifier != null) { 211 | if (this.getIndex(identifier) !== -1) { 212 | throw new Error( 213 | `An item with the identifier ${identifier} already exists`, 214 | ); 215 | } 216 | if (typeof identifier === 'number') { 217 | this.sequence = Math.max(this.sequence, identifier) + 1; 218 | } 219 | } else { 220 | // @ts-expect-error - For some reason, TS does not accept writing a generic types with the index signature 221 | clone[this.identifierName] = this.getNewId(); 222 | } 223 | this.items.push(clone); 224 | return clone; // clone item to avoid returning the original; 225 | } 226 | 227 | updateOne(identifier: number | string, item: T) { 228 | const index = this.getIndex(identifier); 229 | if (index === -1) { 230 | throw new Error(`No item with identifier ${identifier}`); 231 | } 232 | for (const key in item) { 233 | this.items[index][key] = item[key]; 234 | } 235 | return Object.assign({}, this.items[index]); // clone item to avoid returning the original 236 | } 237 | 238 | removeOne(identifier: number | string) { 239 | const index = this.getIndex(identifier); 240 | if (index === -1) { 241 | throw new Error(`No item with identifier ${identifier}`); 242 | } 243 | const item = this.items[index]; 244 | this.items.splice(index, 1); 245 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 246 | if (typeof identifier === 'number' && identifier == this.sequence - 1) { 247 | this.sequence--; 248 | } 249 | return item; 250 | } 251 | } 252 | 253 | const every = ( 254 | array: T[], 255 | predicate: Predicate, 256 | ) => array.reduce((acc, value) => acc && predicate(value), true); 257 | 258 | const some = ( 259 | array: T[], 260 | predicate: Predicate, 261 | ) => array.reduce((acc, value) => acc || predicate(value), false); 262 | 263 | const getArrayOfObjectsPaths = ( 264 | keyParts: string[], 265 | item: T, 266 | ) => 267 | keyParts.reduce( 268 | (acc, key, index) => { 269 | // If we already found an array, we don't need to explore further 270 | // For example with path `tags.name` when tags is an array of objects 271 | if (acc != null) { 272 | return acc; 273 | } 274 | 275 | const keyToArray = keyParts.slice(0, index + 1).join('.'); 276 | const keyToItem = keyParts.slice(index + 1).join('.'); 277 | const itemValue = get(item, keyToArray); 278 | 279 | // If the array is at the end of the key path, we will process it like we do normally with arrays 280 | // For example with path `deep.tags` where tags is the array. In this case, we return undefined 281 | return Array.isArray(itemValue) && index < keyParts.length - 1 282 | ? [keyToArray, keyToItem] 283 | : undefined; 284 | }, 285 | undefined as Array | undefined, 286 | ); 287 | 288 | const getSimpleFilter = (key: string, value: any) => { 289 | if (key.indexOf('_q') !== -1) { 290 | // text search 291 | const realKey = key.replace(/(_q)$/, ''); 292 | const regex = new RegExp(value.toString(), 'i'); 293 | 294 | return (item: T) => 295 | get(item, realKey)?.toString().match(regex) !== null; 296 | } 297 | if (key.indexOf('_lte') !== -1) { 298 | // less than or equal 299 | const realKey = key.replace(/(_lte)$/, ''); 300 | return (item: T) => 301 | get(item, realKey) <= value; 302 | } 303 | if (key.indexOf('_gte') !== -1) { 304 | // less than or equal 305 | const realKey = key.replace(/(_gte)$/, ''); 306 | return (item: T) => 307 | get(item, realKey) >= value; 308 | } 309 | if (key.indexOf('_lt') !== -1) { 310 | // less than or equal 311 | const realKey = key.replace(/(_lt)$/, ''); 312 | return (item: T) => 313 | get(item, realKey) < value; 314 | } 315 | if (key.indexOf('_gt') !== -1) { 316 | // less than or equal 317 | const realKey = key.replace(/(_gt)$/, ''); 318 | return (item: T) => 319 | get(item, realKey) > value; 320 | } 321 | if (key.indexOf('_neq_any') !== -1) { 322 | // not equal to any 323 | const realKey = key.replace(/(_neq_any)$/, ''); 324 | const finalValue = Array.isArray(value) ? value : [value]; 325 | return ( 326 | item: T, 327 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 328 | ) => finalValue.every((val) => get(item, realKey) != val); 329 | } 330 | if (key.indexOf('_neq') !== -1) { 331 | // not equal 332 | const realKey = key.replace(/(_neq)$/, ''); 333 | return ( 334 | item: T, 335 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 336 | ) => get(item, realKey) != value; 337 | } 338 | if (key.indexOf('_eq_any') !== -1) { 339 | // equal any 340 | const realKey = key.replace(/(_eq_any)$/, ''); 341 | const finalValue = Array.isArray(value) ? value : [value]; 342 | return ( 343 | item: T, 344 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 345 | ) => finalValue.some((val) => get(item, realKey) == val); 346 | } 347 | if (key.indexOf('_eq') !== -1) { 348 | // equal 349 | const realKey = key.replace(/(_eq)$/, ''); 350 | return ( 351 | item: T, 352 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 353 | ) => get(item, realKey) == value; 354 | } 355 | if (key.indexOf('_inc_any') !== -1) { 356 | // include any 357 | const realKey = key.replace(/(_inc_any)$/, ''); 358 | const finalValue = Array.isArray(value) ? value : [value]; 359 | return (item: T) => 360 | finalValue.some((val) => { 361 | const itemValue = get(item, realKey); 362 | if (Array.isArray(itemValue)) { 363 | return itemValue.includes(val); 364 | } 365 | if (typeof itemValue === 'string') { 366 | return itemValue.includes(val); 367 | } 368 | return false; 369 | }); 370 | } 371 | if (key.indexOf('_inc') !== -1) { 372 | // includes all 373 | const realKey = key.replace(/(_inc)$/, ''); 374 | const finalValue = Array.isArray(value) ? value : [value]; 375 | return (item: T) => 376 | finalValue.every((val) => { 377 | const itemValue = get(item, realKey); 378 | if (Array.isArray(itemValue)) { 379 | return itemValue.includes(val); 380 | } 381 | if (typeof itemValue === 'string') { 382 | return itemValue.includes(val); 383 | } 384 | return false; 385 | }); 386 | } 387 | if (key.indexOf('_ninc_any') !== -1) { 388 | // does not include any 389 | const realKey = key.replace(/(_ninc_any)$/, ''); 390 | const finalValue = Array.isArray(value) ? value : [value]; 391 | return (item: T) => 392 | finalValue.every((val) => { 393 | const itemValue = get(item, realKey); 394 | if (Array.isArray(itemValue)) { 395 | return !itemValue.includes(val); 396 | } 397 | if (typeof itemValue === 'string') { 398 | return !itemValue.includes(val); 399 | } 400 | return false; 401 | }); 402 | } 403 | if (Array.isArray(value)) { 404 | return (item: T) => { 405 | if (Array.isArray(get(item, key))) { 406 | // array filter and array item value: where all items in values 407 | return every(value, (v) => { 408 | const itemValue = get(item, key); 409 | if (Array.isArray(itemValue)) { 410 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 411 | return some(itemValue, (itemValue) => itemValue == v); 412 | } 413 | return false; 414 | }); 415 | } 416 | // where item in values 417 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 418 | return value.filter((v) => v == get(item, key)).length > 0; 419 | }; 420 | } 421 | 422 | if (value == null) { 423 | // null or undefined filter 424 | return (item: T) => 425 | get(item, key) == null; 426 | } 427 | 428 | if (typeof value === 'object') { 429 | return (item: T) => 430 | matches(value)(get(item, key)); 431 | } 432 | 433 | return (item: T) => { 434 | const itemValue = get(item, key); 435 | if (Array.isArray(itemValue) && typeof value === 'string') { 436 | // simple filter but array item value: where value in item 437 | return itemValue.indexOf(value) !== -1; 438 | } 439 | if (typeof itemValue === 'boolean' && typeof value === 'string') { 440 | // simple filter but boolean item value: boolean where 441 | return itemValue === (value === 'true'); 442 | } 443 | // simple filter 444 | // biome-ignore lint/suspicious/noDoubleEquals: we want implicit type coercion 445 | return itemValue == value; 446 | }; 447 | }; 448 | 449 | function filterItems( 450 | items: T[], 451 | filter: Filter, 452 | ) { 453 | if (typeof filter === 'function') { 454 | return items.filter(filter); 455 | } 456 | if (filter instanceof Object) { 457 | // turn filter properties to functions 458 | const filterFunctions = Object.keys(filter).map((key) => { 459 | if (key === 'q' && typeof filter.q === 'string') { 460 | const regex = buildRegexSearch(filter.q); 461 | 462 | const filterWithQuery = < 463 | T2 extends CollectionItem = CollectionItem, 464 | >( 465 | item: T2, 466 | ) => { 467 | for (const itemKey in item) { 468 | const itemValue = item[itemKey]; 469 | if (typeof itemValue === 'object') { 470 | if (filterWithQuery(itemValue as CollectionItem)) { 471 | return true; 472 | } 473 | } 474 | 475 | if ( 476 | itemValue && 477 | typeof itemValue === 'string' && 478 | itemValue.match && 479 | itemValue.match(regex) !== null 480 | ) 481 | return true; 482 | } 483 | return false; 484 | }; 485 | // full-text filter 486 | return filterWithQuery; 487 | } 488 | 489 | const keyParts = key.split('.'); 490 | const value = filter[key]; 491 | if (keyParts.length > 1) { 492 | return ( 493 | item: T2, 494 | ): boolean => { 495 | const arrayOfObjectsPaths = getArrayOfObjectsPaths( 496 | keyParts, 497 | item, 498 | ); 499 | 500 | if (arrayOfObjectsPaths) { 501 | const [arrayPath, valuePath] = arrayOfObjectsPaths; 502 | const itemValue = get(item, arrayPath); 503 | if (Array.isArray(itemValue)) { 504 | // Check wether any item in the array matches the filter 505 | const filteredArrayItems = filterItems(itemValue, { 506 | [valuePath]: value, 507 | }); 508 | return filteredArrayItems.length > 0; 509 | } 510 | return false; 511 | } 512 | return getSimpleFilter(key, value)(item); 513 | }; 514 | } 515 | 516 | return getSimpleFilter(key, value); 517 | }); 518 | // only the items matching all filters functions are in (AND logic) 519 | return items.filter((item) => 520 | filterFunctions.reduce( 521 | (selected, filterFunction) => selected && filterFunction(item), 522 | true, 523 | ), 524 | ); 525 | } 526 | throw new Error('Unsupported filter type'); 527 | } 528 | 529 | function sortItems( 530 | items: T[], 531 | sort: Sort, 532 | ) { 533 | if (typeof sort === 'function') { 534 | return items.sort(sort); 535 | } 536 | if (typeof sort === 'string') { 537 | return items.sort((a, b) => { 538 | const aValue = get(a, sort); 539 | const bValue = get(b, sort); 540 | if (aValue > bValue) { 541 | return 1; 542 | } 543 | if (aValue < bValue) { 544 | return -1; 545 | } 546 | return 0; 547 | }); 548 | } 549 | if (Array.isArray(sort)) { 550 | const key = sort[0]; 551 | const direction = sort[1].toLowerCase() === 'asc' ? 1 : -1; 552 | return items.sort((a: T, b: T) => { 553 | const aValue = get(a, key); 554 | const bValue = get(b, key); 555 | if (aValue > bValue) { 556 | return direction; 557 | } 558 | if (aValue < bValue) { 559 | return -1 * direction; 560 | } 561 | return 0; 562 | }); 563 | } 564 | throw new Error('Unsupported sort type'); 565 | } 566 | 567 | function rangeItems( 568 | items: T[], 569 | range: Range, 570 | ) { 571 | if (Array.isArray(range)) { 572 | return items.slice( 573 | range[0], 574 | range[1] !== undefined ? range[1] + 1 : undefined, 575 | ); 576 | } 577 | throw new Error('Unsupported range type'); 578 | } 579 | 580 | function buildRegexSearch(input: string) { 581 | // Trim the input to remove leading and trailing whitespace 582 | const trimmedInput = input.trim(); 583 | 584 | // Escape special characters in the input to prevent regex injection 585 | const escapedInput = trimmedInput.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); 586 | 587 | // Split the input into words 588 | const words = escapedInput.split(' '); 589 | 590 | // Create a regex pattern to match any of the words 591 | const pattern = words.map((word) => `(${word})`).join('|'); 592 | 593 | // Create a new RegExp object with the pattern, case insensitive 594 | const regex = new RegExp(pattern, 'i'); 595 | 596 | return regex; 597 | } 598 | -------------------------------------------------------------------------------- /src/adapters/SinonAdapter.spec.ts: -------------------------------------------------------------------------------- 1 | import sinon, { type SinonFakeXMLHttpRequest } from 'sinon'; 2 | 3 | import { SinonAdapter } from './SinonAdapter.ts'; 4 | import type { BaseResponse } from '../types.ts'; 5 | 6 | function getFakeXMLHTTPRequest( 7 | method: string, 8 | url: string, 9 | data?: any, 10 | ): SinonFakeXMLHttpRequest | null { 11 | const xhr = sinon.useFakeXMLHttpRequest(); 12 | let request: SinonFakeXMLHttpRequest | null = null; 13 | xhr.onCreate = (xhr) => { 14 | request = xhr; 15 | }; 16 | const myRequest = new XMLHttpRequest(); 17 | myRequest.open(method, url, false); 18 | myRequest.send(data); 19 | xhr.restore(); 20 | return request; 21 | } 22 | 23 | describe('SinonServer', () => { 24 | describe('addMiddleware', () => { 25 | it('should allow request transformation', async () => { 26 | const server = new SinonAdapter({ 27 | data: { 28 | foo: [ 29 | { id: 1, name: 'foo' }, 30 | { id: 2, name: 'bar' }, 31 | ], 32 | }, 33 | middlewares: [ 34 | (context, next) => { 35 | const start = context.params?._start 36 | ? context.params._start - 1 37 | : 0; 38 | const end = 39 | context.params?._end !== undefined 40 | ? context.params._end - 1 41 | : 19; 42 | if (!context.params) { 43 | context.params = {}; 44 | } 45 | context.params.range = [start, end]; 46 | return next(context); 47 | }, 48 | ], 49 | }); 50 | const handle = server.getHandler(); 51 | let request: SinonFakeXMLHttpRequest | null; 52 | request = getFakeXMLHTTPRequest('GET', '/foo?_start=1&_end=1'); 53 | if (request == null) throw new Error('request is null'); 54 | await handle(request); 55 | expect(request?.status).toEqual(206); 56 | // @ts-ignore 57 | expect(request.responseText).toEqual('[{"id":1,"name":"foo"}]'); 58 | expect(request?.getResponseHeader('Content-Range')).toEqual( 59 | 'items 0-0/2', 60 | ); 61 | request = getFakeXMLHTTPRequest('GET', '/foo?_start=2&_end=2'); 62 | if (request == null) throw new Error('request is null'); 63 | await handle(request); 64 | expect(request?.status).toEqual(206); 65 | // @ts-ignore 66 | expect(request?.responseText).toEqual('[{"id":2,"name":"bar"}]'); 67 | expect(request?.getResponseHeader('Content-Range')).toEqual( 68 | 'items 1-1/2', 69 | ); 70 | }); 71 | 72 | it('should allow response transformation', async () => { 73 | const server = new SinonAdapter({ 74 | data: { 75 | foo: [ 76 | { id: 1, name: 'foo' }, 77 | { id: 2, name: 'bar' }, 78 | ], 79 | }, 80 | middlewares: [ 81 | (context, next) => { 82 | const response = next(context); 83 | (response as BaseResponse).status = 418; 84 | return response; 85 | }, 86 | (context, next) => { 87 | const response = next(context) as BaseResponse; 88 | response.body = { 89 | data: response.body, 90 | status: response.status, 91 | }; 92 | return response; 93 | }, 94 | ], 95 | }); 96 | const handle = server.getHandler(); 97 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 98 | if (request == null) throw new Error('request is null'); 99 | await handle(request); 100 | expect(request.status).toEqual(418); 101 | // @ts-ignore 102 | expect(request.responseText).toEqual( 103 | '{"data":[{"id":1,"name":"foo"},{"id":2,"name":"bar"}],"status":200}', 104 | ); 105 | }); 106 | }); 107 | 108 | describe('handle', () => { 109 | it('should respond a 404 to GET /whatever on non existing collection', async () => { 110 | const server = new SinonAdapter(); 111 | const handle = server.getHandler(); 112 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 113 | if (request == null) throw new Error('request is null'); 114 | await handle(request); 115 | expect(request.status).toEqual(404); // not responded 116 | }); 117 | 118 | it('should respond to GET /foo by sending all items in collection foo', async () => { 119 | const server = new SinonAdapter({ 120 | data: { 121 | foo: [ 122 | { id: 1, name: 'foo' }, 123 | { id: 2, name: 'bar' }, 124 | ], 125 | }, 126 | }); 127 | const handle = server.getHandler(); 128 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 129 | if (request == null) throw new Error('request is null'); 130 | await handle(request); 131 | expect(request.status).toEqual(200); 132 | // @ts-ignore 133 | expect(request.responseText).toEqual( 134 | '[{"id":1,"name":"foo"},{"id":2,"name":"bar"}]', 135 | ); 136 | expect(request.getResponseHeader('Content-Type')).toEqual( 137 | 'application/json', 138 | ); 139 | expect(request.getResponseHeader('Content-Range')).toEqual( 140 | 'items 0-1/2', 141 | ); 142 | }); 143 | 144 | it('should respond to GET /foo?queryString by sending all items in collection foo satisfying query', async () => { 145 | const server = new SinonAdapter({ 146 | data: { 147 | foos: [ 148 | { id: 0, name: 'c', arg: false }, 149 | { id: 1, name: 'b', arg: true }, 150 | { id: 2, name: 'a', arg: true }, 151 | ], 152 | bars: [{ id: 0, name: 'a', foo_id: 1 }], 153 | }, 154 | }); 155 | const handle = server.getHandler(); 156 | const request = getFakeXMLHTTPRequest( 157 | 'GET', 158 | '/foos?filter={"arg":true}&sort=name&slice=[0,10]&embed=["bars"]', 159 | ); 160 | if (request == null) throw new Error('request is null'); 161 | await handle(request); 162 | expect(request.status).toEqual(200); 163 | // @ts-ignore 164 | expect(request.responseText).toEqual( 165 | '[{"id":2,"name":"a","arg":true,"bars":[]},{"id":1,"name":"b","arg":true,"bars":[{"id":0,"name":"a","foo_id":1}]}]', 166 | ); 167 | expect(request.getResponseHeader('Content-Type')).toEqual( 168 | 'application/json', 169 | ); 170 | expect(request.getResponseHeader('Content-Range')).toEqual( 171 | 'items 0-1/2', 172 | ); 173 | }); 174 | 175 | it('should respond to GET /foo?queryString with pagination by sending the correct content-range header', async () => { 176 | const server = new SinonAdapter({ 177 | data: { foo: [{}, {}, {}, {}, {}, {}, {}, {}, {}, {}, {}] }, // 11 items 178 | }); 179 | const handle = server.getHandler(); 180 | let request: SinonFakeXMLHttpRequest | null; 181 | request = getFakeXMLHTTPRequest('GET', '/foo'); 182 | if (request == null) throw new Error('request is null'); 183 | await handle(request); 184 | expect(request.status).toEqual(200); 185 | expect(request.getResponseHeader('Content-Range')).toEqual( 186 | 'items 0-10/11', 187 | ); 188 | request = getFakeXMLHTTPRequest('GET', '/foo?range=[0,4]'); 189 | if (request == null) throw new Error('request is null'); 190 | await handle(request); 191 | expect(request.status).toEqual(206); 192 | expect(request.getResponseHeader('Content-Range')).toEqual( 193 | 'items 0-4/11', 194 | ); 195 | request = getFakeXMLHTTPRequest('GET', '/foo?range=[5,9]'); 196 | if (request == null) throw new Error('request is null'); 197 | await handle(request); 198 | expect(request.status).toEqual(206); 199 | expect(request.getResponseHeader('Content-Range')).toEqual( 200 | 'items 5-9/11', 201 | ); 202 | request = getFakeXMLHTTPRequest('GET', '/foo?range=[10,14]'); 203 | if (request == null) throw new Error('request is null'); 204 | await handle(request); 205 | expect(request.status).toEqual(206); 206 | expect(request.getResponseHeader('Content-Range')).toEqual( 207 | 'items 10-10/11', 208 | ); 209 | }); 210 | 211 | it('should respond to GET /foo on an empty collection with a []', async () => { 212 | const server = new SinonAdapter({ 213 | data: { foo: [] }, 214 | }); 215 | const handle = server.getHandler(); 216 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 217 | if (request == null) throw new Error('request is null'); 218 | await handle(request); 219 | expect(request.status).toEqual(200); 220 | // @ts-ignore 221 | expect(request.responseText).toEqual('[]'); 222 | expect(request.getResponseHeader('Content-Range')).toEqual( 223 | 'items */0', 224 | ); 225 | }); 226 | 227 | it('should respond to POST /foo by adding an item to collection foo', async () => { 228 | const server = new SinonAdapter({ 229 | data: { 230 | foo: [ 231 | { id: 1, name: 'foo' }, 232 | { id: 2, name: 'bar' }, 233 | ], 234 | }, 235 | }); 236 | const handle = server.getHandler(); 237 | const request = getFakeXMLHTTPRequest( 238 | 'POST', 239 | '/foo', 240 | JSON.stringify({ name: 'baz' }), 241 | ); 242 | if (request == null) throw new Error('request is null'); 243 | await handle(request); 244 | expect(request.status).toEqual(201); 245 | // @ts-ignore 246 | expect(request.responseText).toEqual('{"name":"baz","id":3}'); 247 | expect(request.getResponseHeader('Content-Type')).toEqual( 248 | 'application/json', 249 | ); 250 | expect(request.getResponseHeader('Location')).toEqual('/foo/3'); 251 | // @ts-ignore 252 | expect(server.server.database.getAll('foo')).toEqual([ 253 | { id: 1, name: 'foo' }, 254 | { id: 2, name: 'bar' }, 255 | { id: 3, name: 'baz' }, 256 | ]); 257 | }); 258 | 259 | it('should respond to POST /foo by adding an item to collection foo, even if the collection does not exist', async () => { 260 | const server = new SinonAdapter(); 261 | const handle = server.getHandler(); 262 | const request = getFakeXMLHTTPRequest( 263 | 'POST', 264 | '/foo', 265 | JSON.stringify({ name: 'baz' }), 266 | ); 267 | if (request == null) throw new Error('request is null'); 268 | await handle(request); 269 | expect(request.status).toEqual(201); 270 | // @ts-ignore 271 | expect(request.responseText).toEqual('{"name":"baz","id":0}'); 272 | expect(request.getResponseHeader('Content-Type')).toEqual( 273 | 'application/json', 274 | ); 275 | expect(request.getResponseHeader('Location')).toEqual('/foo/0'); 276 | // @ts-ignore 277 | expect(server.server.database.getAll('foo')).toEqual([ 278 | { id: 0, name: 'baz' }, 279 | ]); 280 | }); 281 | 282 | it('should respond to GET /foo/:id by sending element of identifier id in collection foo', async () => { 283 | const server = new SinonAdapter({ 284 | data: { 285 | foo: [ 286 | { id: 1, name: 'foo' }, 287 | { id: 2, name: 'bar' }, 288 | ], 289 | }, 290 | }); 291 | const handle = server.getHandler(); 292 | const request = getFakeXMLHTTPRequest('GET', '/foo/2'); 293 | if (request == null) throw new Error('request is null'); 294 | await handle(request); 295 | expect(request.status).toEqual(200); 296 | // @ts-ignore 297 | expect(request.responseText).toEqual('{"id":2,"name":"bar"}'); 298 | expect(request.getResponseHeader('Content-Type')).toEqual( 299 | 'application/json', 300 | ); 301 | }); 302 | 303 | it('should respond to GET /foo/:id on a non-existing id with a 404', async () => { 304 | const server = new SinonAdapter({ data: { foo: [] } }); 305 | const handle = server.getHandler(); 306 | const request = getFakeXMLHTTPRequest('GET', '/foo/3'); 307 | if (request == null) throw new Error('request is null'); 308 | await handle(request); 309 | expect(request.status).toEqual(404); 310 | }); 311 | 312 | it('should respond to PUT /foo/:id by updating element of identifier id in collection foo', async () => { 313 | const server = new SinonAdapter({ 314 | data: { 315 | foo: [ 316 | { id: 1, name: 'foo' }, 317 | { id: 2, name: 'bar' }, 318 | ], 319 | }, 320 | }); 321 | const handle = server.getHandler(); 322 | const request = getFakeXMLHTTPRequest( 323 | 'PUT', 324 | '/foo/2', 325 | JSON.stringify({ name: 'baz' }), 326 | ); 327 | if (request == null) throw new Error('request is null'); 328 | await handle(request); 329 | expect(request.status).toEqual(200); 330 | // @ts-ignore 331 | expect(request.responseText).toEqual('{"id":2,"name":"baz"}'); 332 | expect(request.getResponseHeader('Content-Type')).toEqual( 333 | 'application/json', 334 | ); 335 | // @ts-ignore 336 | expect(server.server.database.getAll('foo')).toEqual([ 337 | { id: 1, name: 'foo' }, 338 | { id: 2, name: 'baz' }, 339 | ]); 340 | }); 341 | 342 | it('should respond to PUT /foo/:id on a non-existing id with a 404', async () => { 343 | const server = new SinonAdapter(); 344 | const handle = server.getHandler(); 345 | const request = getFakeXMLHTTPRequest( 346 | 'PUT', 347 | '/foo/3', 348 | JSON.stringify({ name: 'baz' }), 349 | ); 350 | if (request == null) throw new Error('request is null'); 351 | await handle(request); 352 | expect(request.status).toEqual(404); 353 | }); 354 | 355 | it('should respond to PATCH /foo/:id by updating element of identifier id in collection foo', async () => { 356 | const server = new SinonAdapter({ 357 | data: { 358 | foo: [ 359 | { id: 1, name: 'foo' }, 360 | { id: 2, name: 'bar' }, 361 | ], 362 | }, 363 | }); 364 | const handle = server.getHandler(); 365 | const request = getFakeXMLHTTPRequest( 366 | 'PATCH', 367 | '/foo/2', 368 | JSON.stringify({ name: 'baz' }), 369 | ); 370 | if (request == null) throw new Error('request is null'); 371 | await handle(request); 372 | expect(request.status).toEqual(200); 373 | // @ts-ignore 374 | expect(request.responseText).toEqual('{"id":2,"name":"baz"}'); 375 | expect(request.getResponseHeader('Content-Type')).toEqual( 376 | 'application/json', 377 | ); 378 | // @ts-ignore 379 | expect(server.server.database.getAll('foo')).toEqual([ 380 | { id: 1, name: 'foo' }, 381 | { id: 2, name: 'baz' }, 382 | ]); 383 | }); 384 | 385 | it('should respond to PATCH /foo/:id on a non-existing id with a 404', async () => { 386 | const server = new SinonAdapter({ data: { foo: [] } }); 387 | const handle = server.getHandler(); 388 | const request = getFakeXMLHTTPRequest( 389 | 'PATCH', 390 | '/foo/3', 391 | JSON.stringify({ name: 'baz' }), 392 | ); 393 | if (request == null) throw new Error('request is null'); 394 | await handle(request); 395 | expect(request.status).toEqual(404); 396 | }); 397 | 398 | it('should respond to DELETE /foo/:id by removing element of identifier id in collection foo', async () => { 399 | const server = new SinonAdapter({ 400 | data: { 401 | foo: [ 402 | { id: 1, name: 'foo' }, 403 | { id: 2, name: 'bar' }, 404 | ], 405 | }, 406 | }); 407 | const handle = server.getHandler(); 408 | const request = getFakeXMLHTTPRequest('DELETE', '/foo/2'); 409 | if (request == null) throw new Error('request is null'); 410 | await handle(request); 411 | expect(request.status).toEqual(200); 412 | // @ts-ignore 413 | expect(request.responseText).toEqual('{"id":2,"name":"bar"}'); 414 | expect(request.getResponseHeader('Content-Type')).toEqual( 415 | 'application/json', 416 | ); 417 | // @ts-ignore 418 | expect(server.server.database.getAll('foo')).toEqual([ 419 | { id: 1, name: 'foo' }, 420 | ]); 421 | }); 422 | 423 | it('should respond to DELETE /foo/:id on a non-existing id with a 404', async () => { 424 | const server = new SinonAdapter({ data: { foo: [] } }); 425 | const handle = server.getHandler(); 426 | const request = getFakeXMLHTTPRequest('DELETE', '/foo/3'); 427 | if (request == null) throw new Error('request is null'); 428 | await handle(request); 429 | expect(request.status).toEqual(404); 430 | }); 431 | 432 | it('should respond to GET /foo/ with single item', async () => { 433 | const server = new SinonAdapter({ 434 | data: { foo: { name: 'foo' } }, 435 | }); 436 | const handle = server.getHandler(); 437 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 438 | if (request == null) throw new Error('request is null'); 439 | await handle(request); 440 | expect(request.status).toEqual(200); 441 | // @ts-ignore 442 | expect(request.responseText).toEqual('{"name":"foo"}'); 443 | expect(request.getResponseHeader('Content-Type')).toEqual( 444 | 'application/json', 445 | ); 446 | }); 447 | 448 | it('should respond to PUT /foo/ by updating the singleton record', async () => { 449 | const server = new SinonAdapter({ 450 | data: { foo: { name: 'foo' } }, 451 | }); 452 | const handle = server.getHandler(); 453 | const request = getFakeXMLHTTPRequest( 454 | 'PUT', 455 | '/foo/', 456 | JSON.stringify({ name: 'baz' }), 457 | ); 458 | if (request == null) throw new Error('request is null'); 459 | await handle(request); 460 | expect(request.status).toEqual(200); 461 | // @ts-ignore 462 | expect(request.responseText).toEqual('{"name":"baz"}'); 463 | expect(request.getResponseHeader('Content-Type')).toEqual( 464 | 'application/json', 465 | ); 466 | // @ts-ignore 467 | expect(server.server.database.getOnly('foo')).toEqual({ 468 | name: 'baz', 469 | }); 470 | }); 471 | 472 | it('should respond to PATCH /foo/ by updating the singleton record', async () => { 473 | const server = new SinonAdapter({ 474 | data: { foo: { name: 'foo' } }, 475 | }); 476 | const handle = server.getHandler(); 477 | const request = getFakeXMLHTTPRequest( 478 | 'PATCH', 479 | '/foo/', 480 | JSON.stringify({ name: 'baz' }), 481 | ); 482 | if (request == null) throw new Error('request is null'); 483 | await handle(request); 484 | expect(request.status).toEqual(200); 485 | // @ts-ignore 486 | expect(request.responseText).toEqual('{"name":"baz"}'); 487 | expect(request.getResponseHeader('Content-Type')).toEqual( 488 | 'application/json', 489 | ); 490 | // @ts-ignore 491 | expect(server.server.database.getOnly('foo')).toEqual({ 492 | name: 'baz', 493 | }); 494 | }); 495 | }); 496 | 497 | describe('setDefaultQuery', () => { 498 | it('should set the default query string', async () => { 499 | const server = new SinonAdapter({ 500 | data: { foo: [{}, {}, {}, {}, {}, {}, {}, {}, {}, {}] }, // 10 items 501 | defaultQuery: () => ({ range: [2, 4] }), 502 | }); 503 | const handle = server.getHandler(); 504 | const request = getFakeXMLHTTPRequest('GET', '/foo'); 505 | if (request == null) throw new Error('request is null'); 506 | await handle(request); 507 | expect(request.status).toEqual(206); 508 | expect(request.getResponseHeader('Content-Range')).toEqual( 509 | 'items 2-4/10', 510 | ); 511 | const expected = [{ id: 2 }, { id: 3 }, { id: 4 }]; 512 | // @ts-ignore 513 | expect(request.responseText).toEqual(JSON.stringify(expected)); 514 | }); 515 | 516 | it('should not override any provided query string', async () => { 517 | const server = new SinonAdapter({ 518 | data: { foo: [{}, {}, {}, {}, {}, {}, {}, {}, {}, {}] }, // 10 items 519 | defaultQuery: () => ({ range: [2, 4] }), 520 | }); 521 | const handle = server.getHandler(); 522 | const request = getFakeXMLHTTPRequest('GET', '/foo?range=[0,4]'); 523 | if (request == null) throw new Error('request is null'); 524 | await handle(request); 525 | expect(request.status).toEqual(206); 526 | expect(request.getResponseHeader('Content-Range')).toEqual( 527 | 'items 0-4/10', 528 | ); 529 | const expected = [ 530 | { id: 0 }, 531 | { id: 1 }, 532 | { id: 2 }, 533 | { id: 3 }, 534 | { id: 4 }, 535 | ]; 536 | // @ts-ignore 537 | expect(request.responseText).toEqual(JSON.stringify(expected)); 538 | }); 539 | }); 540 | }); 541 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # FakeRest 2 | 3 | A browser library that intercepts AJAX calls to mock a REST server based on JSON data. 4 | 5 | Use it in conjunction with [MSW](https://mswjs.io/), [fetch-mock](https://www.wheresrhys.co.uk/fetch-mock/), or [Sinon.js](https://sinonjs.org/releases/v18/fake-xhr-and-server/) to test JavaScript REST clients on the client side (e.g. single page apps) without a server. 6 | 7 | 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)). 8 | 9 | ## Installation 10 | 11 | ```sh 12 | npm install fakerest --save-dev 13 | ``` 14 | 15 | ## Usage 16 | 17 | FakeRest lets you create a handler function that you can pass to an API mocking library. FakeRest supports [MSW](https://mswjs.io/), [fetch-mock](https://www.wheresrhys.co.uk/fetch-mock/), and [Sinon](https://sinonjs.org/releases/v18/fake-xhr-and-server/). If you have the choice, we recommend using MSW, as it will allow you to inspect requests as you usually do in the dev tools network tab. 18 | 19 | ### MSW 20 | 21 | Install [MSW](https://mswjs.io/) and initialize it: 22 | 23 | ```sh 24 | npm install msw@latest --save-dev 25 | npx msw init # eg: public 26 | ``` 27 | 28 | Then configure an MSW worker: 29 | 30 | ```js 31 | // in ./src/fakeServer.js 32 | import { http } from 'msw'; 33 | import { setupWorker } from "msw/browser"; 34 | import { getMswHandler } from "fakerest"; 35 | 36 | const handler = getMswHandler({ 37 | baseUrl: 'http://localhost:3000', 38 | data: { 39 | 'authors': [ 40 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 41 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 42 | ], 43 | 'books': [ 44 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 45 | { id: 1, author_id: 0, title: 'War and Peace' }, 46 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 47 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 48 | ], 49 | 'settings': { 50 | language: 'english', 51 | preferred_format: 'hardback', 52 | } 53 | } 54 | }); 55 | export const worker = setupWorker( 56 | // Make sure you use a RegExp to target all calls to the API 57 | http.all(/http:\/\/localhost:3000/, handler) 58 | ); 59 | ``` 60 | 61 | Finally, call the `worker.start()` method before rendering your application. For instance, in a Vite React application: 62 | 63 | ```js 64 | import React from "react"; 65 | import ReactDom from "react-dom"; 66 | import { App } from "./App"; 67 | import { worker } from "./fakeServer"; 68 | 69 | worker.start({ 70 | quiet: true, // Instruct MSW to not log requests in the console 71 | onUnhandledRequest: 'bypass', // Instruct MSW to ignore requests we don't handle 72 | }).then(() => { 73 | ReactDom.render(, document.getElementById("root")); 74 | }); 75 | ``` 76 | 77 | FakeRest will now intercept every `fetch` request to the REST server. 78 | 79 | ### fetch-mock 80 | 81 | Install [fetch-mock](https://www.wheresrhys.co.uk/fetch-mock/): 82 | 83 | ```sh 84 | npm install fetch-mock --save-dev 85 | ``` 86 | 87 | You can then create a handler and pass it to fetch-mock: 88 | 89 | ```js 90 | import fetchMock from 'fetch-mock'; 91 | import { getFetchMockHandler } from "fakerest"; 92 | 93 | const handler = getFetchMockHandler({ 94 | baseUrl: 'http://localhost:3000', 95 | data: { 96 | 'authors': [ 97 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 98 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 99 | ], 100 | 'books': [ 101 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 102 | { id: 1, author_id: 0, title: 'War and Peace' }, 103 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 104 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 105 | ], 106 | 'settings': { 107 | language: 'english', 108 | preferred_format: 'hardback', 109 | } 110 | } 111 | }); 112 | 113 | fetchMock.mock('begin:http://localhost:3000', handler); 114 | ``` 115 | 116 | FakeRest will now intercept every `fetch` request to the REST server. 117 | 118 | ### Sinon 119 | 120 | Install [Sinon](https://sinonjs.org/releases/v18/fake-xhr-and-server/): 121 | 122 | ```sh 123 | npm install sinon --save-dev 124 | ``` 125 | 126 | Then, configure a Sinon server: 127 | 128 | ```js 129 | import sinon from 'sinon'; 130 | import { getSinonHandler } from "fakerest"; 131 | 132 | const handler = getSinonHandler({ 133 | baseUrl: 'http://localhost:3000', 134 | data: { 135 | 'authors': [ 136 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 137 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 138 | ], 139 | 'books': [ 140 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 141 | { id: 1, author_id: 0, title: 'War and Peace' }, 142 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 143 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 144 | ], 145 | 'settings': { 146 | language: 'english', 147 | preferred_format: 'hardback', 148 | } 149 | }, 150 | }); 151 | 152 | // use sinon.js to monkey-patch XmlHttpRequest 153 | const sinonServer = sinon.fakeServer.create(); 154 | // this is required when doing asynchronous XmlHttpRequest 155 | sinonServer.autoRespond = true; 156 | sinonServer.respondWith(handler); 157 | ``` 158 | 159 | FakeRest will now intercept every `XMLHttpRequest` request to the REST server. 160 | 161 | ## REST Syntax 162 | 163 | FakeRest uses a simple REST syntax described below. 164 | 165 | ### Get A Collection of records 166 | 167 | `GET /[name]` returns an array of records in the `name` collection. It accepts 4 query parameters: `filter`, `sort`, `range`, and `embed`. It responds with a status 200 if there is no pagination, or 206 if the list of items is paginated. The response mentions the total count in the `Content-Range` header. 168 | 169 | GET /books?filter={"author_id":1}&embed=["author"]&sort=["title","desc"]&range=[0-9] 170 | 171 | HTTP 1.1 200 OK 172 | Content-Range: items 0-1/2 173 | Content-Type: application/json 174 | [ 175 | { "id": 3, "author_id": 1, "title": "Sense and Sensibility", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } }, 176 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } } 177 | ] 178 | 179 | The `filter` param must be a serialized object literal describing the criteria to apply to the search query. See the [supported filters](#supported-filters) for more details. 180 | 181 | GET /books?filter={"author_id":1} // return books where author_id is equal to 1 182 | HTTP 1.1 200 OK 183 | Content-Range: items 0-1/2 184 | Content-Type: application/json 185 | [ 186 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" }, 187 | { "id": 3, "author_id": 1, "title": "Sense and Sensibility" } 188 | ] 189 | 190 | // array values are possible 191 | GET /books?filter={"id":[2,3]} // return books where id is in [2,3] 192 | HTTP 1.1 200 OK 193 | Content-Range: items 0-1/2 194 | Content-Type: application/json 195 | [ 196 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" }, 197 | { "id": 3, "author_id": 1, "title": "Sense and Sensibility" } 198 | ] 199 | 200 | // use the special "q" filter to make a full-text search on all text fields 201 | GET /books?filter={"q":"and"} // return books where any of the book properties contains the string 'and' 202 | 203 | HTTP 1.1 200 OK 204 | Content-Range: items 0-2/3 205 | Content-Type: application/json 206 | [ 207 | { "id": 1, "author_id": 0, "title": "War and Peace" }, 208 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" }, 209 | { "id": 3, "author_id": 1, "title": "Sense and Sensibility" } 210 | ] 211 | 212 | // use _gt, _gte, _lte, _lt, or _neq suffix on filter names to make range queries 213 | GET /books?filter={"price_lte":20} // return books where the price is less than or equal to 20 214 | GET /books?filter={"price_gt":20} // return books where the price is greater than 20 215 | 216 | // when the filter object contains more than one property, the criteria combine with an AND logic 217 | GET /books?filter={"published_at_gte":"2015-06-12","published_at_lte":"2015-06-15"} // return books published between two dates 218 | 219 | You can also filter by relationship fields when using embed: 220 | 221 | GET /books?embed=["author"]&filter={"author.name":"Leo Tolstoi"} // return books by Leo Tolstoi 222 | GET /books?embed=["author"]&filter={"author.age_gte":50} // return books by authors aged 50 or more 223 | 224 | The `sort` param must be a serialized array literal defining first the property used for sorting, then the sorting direction. 225 | 226 | GET /author?sort=["date_of_birth","asc"] // return authors, the oldest first 227 | GET /author?sort=["date_of_birth","desc"] // return authors, the youngest first 228 | 229 | You can also sort by relationship fields when using embed: 230 | 231 | GET /books?embed=["author"]&sort=["author.name","asc"] // return books sorted by author name 232 | GET /books?embed=["author"]&sort=["author.name","desc"] // return books sorted by author name in reverse order 233 | 234 | The `range` param defines the number of results by specifying the rank of the first and last results. The first result is #0. 235 | 236 | GET /books?range=[0-9] // return the first 10 books 237 | GET /books?range=[10-19] // return the 10 next books 238 | 239 | The `embed` param sets the related objects or collections to be embedded in the response. 240 | 241 | // embed author in books 242 | GET /books?embed=["author"] 243 | HTTP 1.1 200 OK 244 | Content-Range: items 0-3/4 245 | Content-Type: application/json 246 | [ 247 | { "id": 0, "author_id": 0, "title": "Anna Karenina", "author": { "id": 0, "first_name": "Leo", "last_name": "Tolstoi" } }, 248 | { "id": 1, "author_id": 0, "title": "War and Peace", "author": { "id": 0, "first_name": "Leo", "last_name": "Tolstoi" } }, 249 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } }, 250 | { "id": 3, "author_id": 1, "title": "Sense and Sensibility", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } } 251 | ] 252 | 253 | // embed books in author 254 | GET /authors?embed=["books"] 255 | HTTP 1.1 200 OK 256 | Content-Range: items 0-1/2 257 | Content-Type: application/json 258 | [ 259 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi', books: [{ id: 0, author_id: 0, title: 'Anna Karenina' }, { id: 1, author_id: 0, title: 'War and Peace' }] }, 260 | { id: 1, first_name: 'Jane', last_name: 'Austen', books: [{ id: 2, author_id: 1, title: 'Pride and Prejudice' }, { id: 3, author_id: 1, title: 'Sense and Sensibility' }] } 261 | ] 262 | 263 | // you can embed several objects 264 | GET /authors?embed=["books","country"] 265 | 266 | ### Get A Single Record 267 | 268 | `GET /[name]/:id` returns a JSON object, and a status 200, unless the resource doesn't exist. 269 | 270 | GET /books/2 271 | 272 | HTTP 1.1 200 OK 273 | Content-Type: application/json 274 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" } 275 | 276 | The `embed` param sets the related objects or collections to be embedded in the response. 277 | 278 | GET /books/2?embed=['author'] 279 | 280 | HTTP 1.1 200 OK 281 | Content-Type: application/json 282 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } } 283 | 284 | ### Create A Record 285 | 286 | `POST /[name]` returns a status 201 with a `Location` header for the newly created resource, and the new resource in the body. 287 | 288 | POST /books 289 | { "author_id": 1, "title": "Emma" } 290 | 291 | HTTP 1.1 201 Created 292 | Location: /books/4 293 | Content-Type: application/json 294 | { "author_id": 1, "title": "Emma", "id": 4 } 295 | 296 | ### Update A Record 297 | 298 | `PUT /[name]/:id` returns the modified JSON object, and a status 200, unless the resource doesn't exist. 299 | 300 | PUT /books/2 301 | { "author_id": 1, "title": "Pride and Prejudice" } 302 | 303 | HTTP 1.1 200 OK 304 | Content-Type: application/json 305 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" } 306 | 307 | ### Delete A Single Record 308 | 309 | `DELETE /[name]/:id` returns the deleted JSON object, and a status 200, unless the resource doesn't exist. 310 | 311 | DELETE /books/2 312 | 313 | HTTP 1.1 200 OK 314 | Content-Type: application/json 315 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice" } 316 | 317 | ### Supported Filters 318 | 319 | Operators are specified as suffixes on each filtered field. For instance, applying the `_lte` operator on the `price` field for the `books` resource is done like this: 320 | 321 | GET /books?filter={"price_lte":20} // return books where the price is less than or equal to 20 322 | 323 | - `_eq`: check for equality on simple values: 324 | 325 | GET /books?filter={"price_eq":20} // return books where the price is equal to 20 326 | 327 | - `_neq`: check for inequality on simple values 328 | 329 | GET /books?filter={"price_neq":20} // return books where the price is not equal to 20 330 | 331 | - `_eq_any`: check for equality on any passed values 332 | 333 | GET /books?filter={"price_eq_any":[20, 30]} // return books where the price is equal to 20 or 30 334 | 335 | - `_neq_any`: check for inequality on any passed values 336 | 337 | GET /books?filter={"price_neq_any":[20, 30]} // return books where the price is not equal to 20 nor 30 338 | 339 | - `_inc_any`: check for items that include any of the passed values 340 | 341 | GET /books?filter={"authors_inc_any":['William Gibson', 'Pat Cadigan']} // return books where authors include either 'William Gibson' or 'Pat Cadigan' or both 342 | 343 | - `_q`: check for items that contain the provided text 344 | 345 | GET /books?filter={"author_q":['Gibson']} // return books where the author includes 'Gibson' not considering the other fields 346 | 347 | - `_lt`: check for items that have a value lower than the provided value 348 | 349 | GET /books?filter={"price_lte":100} // return books that have a price lower that 100 350 | 351 | - `_lte`: check for items that have a value lower than or equal to the provided value 352 | 353 | GET /books?filter={"price_lte":100} // return books that have a price lower or equal to 100 354 | 355 | - `_gt`: check for items that have a value greater than the provided value 356 | 357 | GET /books?filter={"price_gte":100} // return books that have a price greater that 100 358 | 359 | - `_gte`: check for items that have a value greater than or equal to the provided value 360 | 361 | GET /books?filter={"price_gte":100} // return books that have a price greater or equal to 100 362 | 363 | ### Single Elements 364 | 365 | FakeRest allows you to define a single element, such as a user profile or global settings, that can be fetched, updated, or deleted. 366 | 367 | GET /settings 368 | 369 | HTTP 1.1 200 OK 370 | Content-Type: application/json 371 | { "language": "english", "preferred_format": "hardback" } 372 | 373 | PUT /settings 374 | { "language": "french", "preferred_format": "paperback" } 375 | 376 | HTTP 1.1 200 OK 377 | Content-Type: application/json 378 | { "language": "french", "preferred_format": "paperback" } 379 | 380 | DELETE /settings 381 | 382 | HTTP 1.1 200 OK 383 | Content-Type: application/json 384 | { "language": "french", "preferred_format": "paperback" } 385 | 386 | ## Middlewares 387 | 388 | Middlewares let you intercept requests and simulate server features such as: 389 | - authentication checks 390 | - server-side validation 391 | - server dynamically generated values 392 | - simulate response delays 393 | 394 | You can define middlewares on all handlers, by passing a `middlewares` option: 395 | 396 | ```js 397 | import { getMswHandler } from 'fakerest'; 398 | import { data } from './data'; 399 | 400 | const handler = getMswHandler({ 401 | baseUrl: 'http://my.custom.domain', 402 | data, 403 | middlewares: [ 404 | async (context, next) => { 405 | if (context.headers.Authorization === undefined) { 406 | return { 407 | status: 401, 408 | headers: {}, 409 | }; 410 | } 411 | 412 | return next(context); 413 | }, 414 | withDelay(300), 415 | ], 416 | }); 417 | ``` 418 | 419 | A middleware is a function that receives 2 parameters: 420 | - The FakeRest `context`, an object containing the data extracted from the request that FakeRest uses to build the response. It has the following properties: 421 | - `method`: The request method as a string (`GET`, `POST`, `PATCH` or `PUT`) 422 | - `url`: The request URL as a string 423 | - `headers`: The request headers as an object where keys are header names 424 | - `requestBody`: The parsed request data if any 425 | - `params`: The request parameters from the URL search (e.g. the identifier of the requested record) 426 | - `collection`: The name of the targeted [collection](#collection) (e.g. `posts`) 427 | - `single`: The name of the targeted [single](#single) (e.g. `settings`) 428 | - A `next` function to call the next middleware in the chain, to which you must pass the `context` 429 | 430 | 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: 431 | - `status`: The response status as a number (e.g. `200`) 432 | - `headers`: The response HTTP headers as an object where keys are header names 433 | - `body`: The response body which will be stringified 434 | 435 | ### Authentication Checks 436 | 437 | Here's how to implement an authentication check: 438 | 439 | ```js 440 | const handler = getMswHandler({ 441 | baseUrl: 'http://my.custom.domain', 442 | data, 443 | middlewares: [ 444 | async (context, next) => { 445 | if (context.headers.Authorization === undefined) { 446 | return { status: 401, headers: {} }; 447 | } 448 | return next(context); 449 | } 450 | ] 451 | }); 452 | ``` 453 | 454 | ### Server-Side Validation 455 | 456 | Here's how to implement server-side validation: 457 | 458 | ```js 459 | const handler = getMswHandler({ 460 | baseUrl: 'http://my.custom.domain', 461 | data, 462 | middlewares: [ 463 | async (context, next) => { 464 | if ( 465 | context.collection === "books" && 466 | request.method === "POST" && 467 | !context.requestBody?.title 468 | ) { 469 | return { 470 | status: 400, 471 | headers: {}, 472 | body: { 473 | errors: { 474 | title: 'An article with this title already exists. The title must be unique.', 475 | }, 476 | }, 477 | }; 478 | } 479 | 480 | return next(context); 481 | } 482 | ] 483 | }); 484 | ``` 485 | 486 | ### Dynamically Generated Values 487 | 488 | Here's how to implement dynamically generated values on creation: 489 | 490 | ```js 491 | const handler = getMswHandler({ 492 | baseUrl: 'http://my.custom.domain', 493 | data, 494 | middlewares: [ 495 | async (context, next) => { 496 | if ( 497 | context.collection === 'books' && 498 | context.method === 'POST' 499 | ) { 500 | const response = await next(context); 501 | response.body.updatedAt = new Date().toISOString(); 502 | return response; 503 | } 504 | 505 | return next(context); 506 | } 507 | ] 508 | }); 509 | ``` 510 | 511 | ### Simulate Response Delays 512 | 513 | Here's how to simulate response delays: 514 | 515 | ```js 516 | const handler = getMswHandler({ 517 | baseUrl: 'http://my.custom.domain', 518 | data, 519 | middlewares: [ 520 | async (context, next) => { 521 | return new Promise((resolve) => { 522 | setTimeout(() => { 523 | resolve(next(context)); 524 | }, 500); 525 | }); 526 | } 527 | ] 528 | }); 529 | ``` 530 | 531 | This is so common FakeRest provides the `withDelay` function for that: 532 | 533 | ```js 534 | import { getMswHandler, withDelay } from 'fakerest'; 535 | 536 | const handler = getMswHandler({ 537 | baseUrl: 'http://my.custom.domain', 538 | data, 539 | middlewares: [ 540 | withDelay(500), // delay in ms 541 | ] 542 | }); 543 | ``` 544 | 545 | ## Configuration 546 | 547 | All handlers can be customized to accommodate your API structure. 548 | 549 | ### Identifiers 550 | 551 | By default, FakeRest assumes all records have a unique `id` field. 552 | Some databases such as [MongoDB](https://www.mongodb.com) use `_id` instead of `id` for collection identifiers. 553 | You can customize FakeRest to do the same by setting the `identifierName` option: 554 | 555 | ```js 556 | const handler = getMswHandler({ 557 | baseUrl: 'http://my.custom.domain', 558 | data, 559 | identifierName: '_id' 560 | }); 561 | ``` 562 | 563 | You can also specify that on a per-collection basis: 564 | 565 | ```js 566 | import { MswAdapter, Collection } from 'fakerest'; 567 | 568 | const adapter = new MswAdapter({ baseUrl: 'http://my.custom.domain', data }); 569 | const authorsCollection = new Collection({ items: [], identifierName: '_id' }); 570 | adapter.server.addCollection('authors', authorsCollection); 571 | const handler = adapter.getHandler(); 572 | ``` 573 | 574 | ### Primary Keys 575 | 576 | By default, FakeRest uses an auto-incremented sequence for the item identifiers. 577 | 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: 578 | 579 | ```js 580 | import { getMswHandler } from 'fakerest'; 581 | import uuid from 'uuid'; 582 | 583 | const handler = new getMswHandler({ 584 | baseUrl: 'http://my.custom.domain', 585 | data, 586 | getNewId: () => uuid.v5() 587 | }); 588 | ``` 589 | 590 | You can also specify that on a per-collection basis: 591 | 592 | ```js 593 | import { MswAdapter, Collection } from 'fakerest'; 594 | import uuid from 'uuid'; 595 | 596 | const adapter = new MswAdapter({ baseUrl: 'http://my.custom.domain', data }); 597 | const authorsCollection = new Collection({ items: [], getNewId: () => uuid.v5() }); 598 | adapter.server.addCollection('authors', authorsCollection); 599 | const handler = adapter.getHandler(); 600 | ``` 601 | 602 | ### Default Queries 603 | 604 | Some APIs might enforce some parameters on queries. For instance, an API might always include an [embed](#embed) or enforce a query filter. 605 | You can simulate this using the `defaultQuery` parameter: 606 | 607 | ```js 608 | import { getMswHandler } from 'fakerest'; 609 | import uuid from 'uuid'; 610 | 611 | const handler = getMswHandler({ 612 | baseUrl: 'http://my.custom.domain', 613 | data, 614 | defaultQuery: (collection) => { 615 | if (resourceName == 'authors') return { embed: ['books'] } 616 | if (resourceName == 'books') return { filter: { published: true } } 617 | return {}; 618 | } 619 | }); 620 | ``` 621 | 622 | ## Architecture 623 | 624 | Behind a simple API (`getXXXHandler`), FakeRest uses a modular architecture that lets you combine different components to build a fake REST server that fits your needs. 625 | 626 | ### Mocking Adapter 627 | 628 | `getXXXHandler` is a shortcut to an object-oriented API of adapter classes: 629 | 630 | ```js 631 | export const getMswHandler = (options: MswAdapterOptions) => { 632 | const server = new MswAdapter(options); 633 | return server.getHandler(); 634 | }; 635 | ``` 636 | 637 | FakeRest provides 3 adapter classes: 638 | 639 | - `MswAdapter`: Based on [MSW](https://mswjs.io/) 640 | - `FetchMockAdapter`: Based on [`fetch-mock`](https://www.wheresrhys.co.uk/fetch-mock/) 641 | - `SinonAdapter`: Based on [Sinon](https://sinonjs.org/releases/v18/fake-xhr-and-server/) 642 | 643 | You can use the adapter class directly, e.g. if you want to make the adapter instance available in the global scope for debugging purposes: 644 | 645 | ```js 646 | import { MsWAdapter } from 'fakerest'; 647 | 648 | const adapter = new MswAdapter({ 649 | baseUrl: 'http://my.custom.domain', 650 | data: { 651 | 'authors': [ 652 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 653 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 654 | ], 655 | 'books': [ 656 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 657 | { id: 1, author_id: 0, title: 'War and Peace' }, 658 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 659 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 660 | ], 661 | 'settings': { 662 | language: 'english', 663 | preferred_format: 'hardback', 664 | } 665 | } 666 | }); 667 | window.fakerest = adapter; 668 | const handler = adapter.getHandler(); 669 | ``` 670 | 671 | ### REST Server 672 | 673 | Adapters transform requests to a normalized format, pass them to a server object, and transform the normalized server response into the format expected by the mocking library. 674 | 675 | The server object implements the REST syntax. It takes a normalized request and exposes a `handle` method that returns a normalized response. FakeRest currently provides only one server implementation: `SimpleRestServer`. 676 | 677 | You can specify the server to use in an adapter by passing the `server` option: 678 | 679 | ```js 680 | const server = new SimpleRestServer({ 681 | baseUrl: 'http://my.custom.domain', 682 | data: { 683 | 'authors': [ 684 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 685 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 686 | ], 687 | 'books': [ 688 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 689 | { id: 1, author_id: 0, title: 'War and Peace' }, 690 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 691 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 692 | ], 693 | 'settings': { 694 | language: 'english', 695 | preferred_format: 'hardback', 696 | } 697 | } 698 | }); 699 | const adapter = new MswAdapter({ server }); 700 | const handler = adapter.getHandler(); 701 | ``` 702 | 703 | You can provide an alternative server implementation. This class must implement the `APIServer` type: 704 | 705 | ```ts 706 | export type APIServer = { 707 | baseUrl?: string; 708 | handle: (context: FakeRestContext) => Promise; 709 | }; 710 | 711 | export type BaseResponse = { 712 | status: number; 713 | body?: Record | Record[]; 714 | headers: { [key: string]: string }; 715 | }; 716 | 717 | export type FakeRestContext = { 718 | url?: string; 719 | headers?: Headers; 720 | method?: string; 721 | collection?: string; 722 | single?: string; 723 | requestBody: Record | undefined; 724 | params: { [key: string]: any }; 725 | }; 726 | ``` 727 | 728 | The `FakerRestContext` type describes the normalized request. It's usually the adapter's job to transform the request from the mocking library to this format. 729 | 730 | ### Database 731 | 732 | The querying logic is implemented in a class called `Database`, which is independent of the server. It contains [collections](#collections) and [single](#single). 733 | 734 | You can specify the database used by a server by setting its `database` property: 735 | 736 | ```js 737 | const database = new Database({ 738 | data: { 739 | 'authors': [ 740 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 741 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 742 | ], 743 | 'books': [ 744 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 745 | { id: 1, author_id: 0, title: 'War and Peace' }, 746 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 747 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 748 | ], 749 | 'settings': { 750 | language: 'english', 751 | preferred_format: 'hardback', 752 | } 753 | } 754 | }); 755 | const server = new SimpleRestServer({ baseUrl: 'http://my.custom.domain', database }); 756 | ``` 757 | 758 | You can even use the database object if you want to manipulate the data: 759 | 760 | ```js 761 | database.updateOne('authors', 0, { first_name: 'Lev' }); 762 | ``` 763 | 764 | ### Collections & Singles 765 | 766 | The Database may contain collections and singles. In the following example, `authors` and `books` are collections, and `settings` is a single. 767 | 768 | ```js 769 | const handler = getMswHandler({ 770 | baseUrl: 'http://localhost:3000', 771 | data: { 772 | 'authors': [ 773 | { id: 0, first_name: 'Leo', last_name: 'Tolstoi' }, 774 | { id: 1, first_name: 'Jane', last_name: 'Austen' } 775 | ], 776 | 'books': [ 777 | { id: 0, author_id: 0, title: 'Anna Karenina' }, 778 | { id: 1, author_id: 0, title: 'War and Peace' }, 779 | { id: 2, author_id: 1, title: 'Pride and Prejudice' }, 780 | { id: 3, author_id: 1, title: 'Sense and Sensibility' } 781 | ], 782 | 'settings': { 783 | language: 'english', 784 | preferred_format: 'hardback', 785 | } 786 | } 787 | }); 788 | ``` 789 | 790 | A collection is the equivalent of a classic database table. It supports filtering and direct access to records by their identifier. 791 | 792 | A single represents an API endpoint that returns a single entity. It's useful for things such as user profile routes (`/me`) or global settings (`/settings`). 793 | 794 | ### Embeds 795 | 796 | FakeRest supports embedding other resources in a main resource query result. For instance, embedding the author of a book. 797 | 798 | GET /books/2?embed=['author'] 799 | 800 | HTTP 1.1 200 OK 801 | Content-Type: application/json 802 | { "id": 2, "author_id": 1, "title": "Pride and Prejudice", "author": { "id": 1, "first_name": "Jane", "last_name": "Austen" } } 803 | 804 | Embeds are defined by the query, they require no setup in the database. 805 | 806 | ## Development 807 | 808 | ```sh 809 | # Install dependencies 810 | make install 811 | 812 | # Run the demo with MSW 813 | make run-msw 814 | 815 | # Run the demo with fetch-mock 816 | make run-fetch-mock 817 | 818 | # Run the demo with sinon 819 | make run-sinon 820 | 821 | # Run tests 822 | make test 823 | 824 | # Build minified version 825 | make build 826 | ``` 827 | 828 | You can sign-in to the demo with `janedoe` and `password` 829 | 830 | ## License 831 | 832 | FakeRest is licensed under the [MIT License](LICENSE), sponsored by [marmelab](http://marmelab.com). 833 | --------------------------------------------------------------------------------