Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

feat: support cross-process interception via setupRemoteServer #1617

Open
wants to merge 252 commits into
base: main
Choose a base branch
from
Open
Changes from 1 commit
Commits
Show all changes
252 commits
Select commit Hold shift + click to select a range
90dd65e
feat: support "HttpResponse.arrayBuffer()"
kettanaito Oct 8, 2022
49369e6
fix(HttpResponse): forward cookies in the browser only
kettanaito Oct 8, 2022
3b947c7
docs: update readme
kettanaito Oct 8, 2022
b909e48
fix(pruneGetRequestBody): support HEAD requests
kettanaito Oct 9, 2022
f9840b7
chore: polyfill Request in jest tests
kettanaito Oct 9, 2022
b87724d
feat: support "bypass" utility
kettanaito Oct 9, 2022
3315cd4
chore: update "headers-polyfill" to 3.1.2
kettanaito Oct 9, 2022
aef1db1
test(headers-multiple): separate headers by comma
kettanaito Oct 9, 2022
b44a993
feat: support new api for graphql
kettanaito Oct 9, 2022
71753c3
chore: migrate "msw-api" tests
kettanaito Oct 10, 2022
a5a91dc
feat: add "delay()"
kettanaito Oct 10, 2022
0ee92d8
feat: migrate to new interceptors
kettanaito Oct 14, 2022
7fcadf2
fix(RequestHandler): clone response in generator
kettanaito Oct 14, 2022
d64713a
chore: migrate to fetch api in node
kettanaito Oct 14, 2022
fd89534
feat: export universal fetch classes
kettanaito Oct 14, 2022
5166b5e
fix(bypass): use compatible Request type
kettanaito Oct 15, 2022
11fa7c6
feat: support one-time request handlers
kettanaito Oct 15, 2022
adab915
feat: add "passthrough", fix tests
kettanaito Oct 15, 2022
dde933b
feat: support multi-value response cookies
kettanaito Oct 16, 2022
13bb61f
chore: add "File" polyfill for node
kettanaito Oct 16, 2022
ca158f3
feat: add "HttpResponse.formData()"
kettanaito Oct 16, 2022
e494683
test: fix request FormData body test
kettanaito Oct 16, 2022
41ffaa0
chore: remove unused code
kettanaito Oct 17, 2022
547eecb
docs: add the migration guide
kettanaito Oct 17, 2022
e8fca8b
feat: support response body type generic
kettanaito Oct 17, 2022
6771bb3
fix(HttpResponse): accept "string" as input to ".json()"
kettanaito Oct 17, 2022
a4adce0
feat: support strict request body type
kettanaito Oct 17, 2022
490d2b8
test: add path params types tests
kettanaito Oct 17, 2022
62a7289
feat(graphql): support strict response body type
kettanaito Oct 17, 2022
7a56a7d
feat: support mock ReadableStream responses
kettanaito Oct 18, 2022
f3ee2f4
feat: handle request body as ArrayBuffer
kettanaito Oct 18, 2022
896ae0e
test: fix graphql typings tests
kettanaito Oct 18, 2022
f78175f
test: remove "set" typings tests
kettanaito Oct 18, 2022
b84fd6b
chore: remove unused "compose"
kettanaito Oct 18, 2022
533f44c
chore(NetworkError): add jsdoc
kettanaito Oct 18, 2022
b562f1f
fix(GraphQLHandler): annotate "errors" as partial
kettanaito Oct 18, 2022
6fd2eb8
chore(HttpResponse): remove unnecessary "defineReadOnly"
kettanaito Oct 18, 2022
6f07723
chore: move "HttpResponse" to the root
kettanaito Oct 18, 2022
d61a5db
test: add unit tests for HttpResponse
kettanaito Oct 18, 2022
801b5ae
chore: move fetch polyfills to root
kettanaito Oct 18, 2022
24a59e5
fix: correct types
kettanaito Oct 19, 2022
de0bd2f
fix(RequestHandler): set ResponseBodyType as undefined by default
kettanaito Oct 19, 2022
a27f6b7
test(cookies-inheritance): use a response body union type
kettanaito Oct 19, 2022
e005153
fix(bypass): support request generic to satisfy various polyfills
kettanaito Oct 21, 2022
03e6658
Merge branch 'main' into feat/standard-api
kettanaito Oct 21, 2022
f3894cf
chore: update to @mswjs/[email protected]
kettanaito Nov 7, 2022
0157d82
chore: use @swc/jest vs ts-jest
kettanaito Nov 8, 2022
9b50319
fix(bypass): return [url, init] tuple for polyfill compatibility
kettanaito Nov 8, 2022
895e8e6
Merge branch 'main' into feat/standard-api
kettanaito Nov 14, 2022
10e3691
chore: fix failing bypass test
kettanaito Nov 14, 2022
8874812
chore: fix worker context event listener compatibility
kettanaito Nov 14, 2022
6861097
fix: typescript <= 4.4 compatibility
kettanaito Nov 14, 2022
b14380b
docs: improve migration guide
kettanaito Nov 15, 2022
b320fe5
test: fix response-patching test
kettanaito Nov 15, 2022
2684382
fix(bypass): forward request body
kettanaito Nov 15, 2022
085d52b
test: fix failing tests
kettanaito Nov 15, 2022
4edf17f
Merge branch 'main' into feat/standard-api
kettanaito Nov 15, 2022
77ca45a
Merge branch 'main' into feat/standard-api
kettanaito Nov 15, 2022
d84b2d0
chore: v0.0.0-fetch.rc-1
kettanaito Nov 15, 2022
b48b42e
Merge branch 'main' into feat/standard-api
kettanaito Nov 16, 2022
09218f3
docs(migrating): remove duplicate handler in set-cookies
kettanaito Nov 16, 2022
84978a0
docs: use HeadersInit for multi-value cookies
kettanaito Nov 17, 2022
3875d9b
chore(HttpResponse): remove commented code
kettanaito Nov 17, 2022
80ed635
docs: fix typo in migration guide (#1469)
Xayer Nov 18, 2022
cb2766d
Merge branch 'main' into feat/standard-api
kettanaito Nov 24, 2022
041ff3e
chore: v0.0.0-fetch.rc-2
kettanaito Nov 24, 2022
2f8bf27
fix(HttpResponse): accept interfaces as json response-data type (#1481)
christoph-fricke Nov 28, 2022
4bea835
docs(migrating): fix example for response composition (#1483)
christoph-fricke Dec 2, 2022
3a001af
docs(migrating): fix description for "ctx.errors" migration (#1487)
christoph-fricke Dec 2, 2022
7788593
Merge branch 'main' into feat/standard-api
kettanaito Dec 6, 2022
ffc51db
Merge branch 'main' into feat/standard-api
kettanaito Dec 7, 2022
9ba34e1
chore(release): v0.0.0-fetch.rc-3
kettanaito Dec 7, 2022
9607880
Merge branch 'main' into feat/standard-api
kettanaito Jan 11, 2023
6b87f2b
fix(decorateResponseInit): use global "Headers" if available
kettanaito Jan 11, 2023
d1e82e7
Merge branch 'main' into feat/standard-api
kettanaito Jan 14, 2023
cff6452
fix: update to @mswjs/[email protected]
kettanaito Jan 14, 2023
021a080
Merge branch 'main' into feat/standard-api
kettanaito Jan 27, 2023
1bfd751
fix(worker): remove bypassing of server-sent events (#1551)
piotr-cz Feb 21, 2023
9a2446b
feat: drop support for node < 18
kettanaito Feb 27, 2023
b614531
Merge branch 'main' into feat/standard-api
kettanaito Feb 27, 2023
0fd5019
chore: fix unit tests
kettanaito Feb 27, 2023
d0a27ef
test(xhr): rewrite to fetch api
kettanaito Feb 27, 2023
7062869
chore: fix node tests
kettanaito Mar 1, 2023
603b24a
chore: use node 18 on ci
kettanaito Mar 3, 2023
e39b8fe
chore: move interceptors back to regular dependencies
kettanaito Mar 3, 2023
7acbf21
test: fix rest-api body mocks test
kettanaito Mar 3, 2023
fc38370
chore: fix browser tests
kettanaito Mar 3, 2023
69d2c65
chore(xhr): fix onload being called twice in tests
kettanaito Mar 4, 2023
5e92590
chore(release): v0.0.0-fetch.rc-4
kettanaito Mar 4, 2023
2273782
chore(release): v0.0.0-fetch.rc-5
kettanaito Mar 4, 2023
22cb9a1
feat: distribute library as cjs and esm
kettanaito Mar 7, 2023
c0c7f2a
fix: set "main" to node, use explicit "browser" field
kettanaito Mar 7, 2023
e4dce71
chore(release): 0.0.0-fetch.rc-6
kettanaito Mar 7, 2023
694bec9
test(on): clone response before reading it
kettanaito Mar 7, 2023
0f08330
chore: use node v18.14.2 internally
kettanaito Mar 7, 2023
a9cd280
test: unskip form-data node test
kettanaito Mar 7, 2023
e2c6d5a
Merge branch 'main' into feat/standard-api
kettanaito Mar 21, 2023
a369914
fix: update "strict-event-emitter" to 0.5.0
kettanaito Mar 21, 2023
37617c8
chore(release): v0.0.0-fetch.rc-7
kettanaito Mar 21, 2023
c347a9e
Merge branch 'main' into feat/standard-api
kettanaito Mar 23, 2023
2d57879
Merge branch 'main' into feat/standard-api
kettanaito Mar 24, 2023
bedcb2c
chore(release): v0.0.0-fetch.rc-8
kettanaito Mar 24, 2023
c4367a5
fix: migrate to "@open-draft/until@2"
kettanaito Mar 30, 2023
c5d6600
chore: update to "@mswjs/[email protected]"
kettanaito Mar 30, 2023
472a358
chore: update to "[email protected]"
kettanaito Mar 30, 2023
14cf7d7
Merge branch 'main' into feat/standard-api
kettanaito Apr 6, 2023
7c6c9d9
feat: split the library into "core", "browser" and "node" (#1584)
kettanaito Apr 11, 2023
651e1a8
feat: export browser integration from "msw/browser" (#1591)
kettanaito Apr 11, 2023
206682f
docs: mention "msw/browser" change in migration guidelines
kettanaito Apr 11, 2023
f5637d1
chore(release): v0.0.0-fetch.rc-9
kettanaito Apr 11, 2023
34ae9da
fix: include "browser" in "files"
kettanaito Apr 12, 2023
e7153d9
chore(release): 0.0.0-fetch.rc-10
kettanaito Apr 12, 2023
76bac29
docs: add missing quote to migration guidelines (#1592)
Apr 12, 2023
6af4b3c
fix: ditch "debug" in favor of "@open-draft/logger"
kettanaito Apr 13, 2023
f4b4b4f
fix: use "statuses" plain object, list as dependency
kettanaito Apr 13, 2023
f656f0c
fix: bump to interceptors 0.22.12
kettanaito Apr 14, 2023
b8376e4
chore: explain "customConditions" in jest (jsdom)
kettanaito Apr 14, 2023
a4dadc9
chore: force exit node tests
kettanaito Apr 14, 2023
b816698
chore(release): v0.0.0-fetch.rc-11
kettanaito Apr 14, 2023
2331179
chore: add automated modules tests
kettanaito Apr 17, 2023
e48f435
chore: add runtime esm node test
kettanaito Apr 17, 2023
99402f6
fix: drop "chalk"
kettanaito Apr 17, 2023
1f81582
fix: use default import for cjs "statuses"
kettanaito Apr 17, 2023
0074b52
test: add runtime esm node.js tests
kettanaito Apr 17, 2023
a3b5c09
test: fix printHandlers location tests
kettanaito Apr 17, 2023
4c21110
chore: add browser module tests
kettanaito Apr 17, 2023
0fd26dc
chore: use "@web/dev-server" for browser esm tests
kettanaito Apr 18, 2023
4f3e05b
chore: update in-house dependencies for esm compat
kettanaito Apr 18, 2023
8ae5a4f
chore: use correct cjs deps imports
kettanaito Apr 18, 2023
f1749a6
chore: fix typescript package import in ts tests
kettanaito Apr 18, 2023
c7139df
fix(cookie): use manually bundled esm version
kettanaito Apr 20, 2023
9b27a12
chore: temporarily fix "process.env" for "graphql"
kettanaito Apr 25, 2023
7ff17ae
feat(HttpResponse): make initializable class
kettanaito Apr 28, 2023
0f35fd2
fix: esm fixes
thepassle Apr 29, 2023
100a43f
fix: annotate "@bundled-es-modules/statuses"
kettanaito Apr 29, 2023
94ec2ca
Merge branch 'main' into feat/standard-api
kettanaito Apr 29, 2023
36a0dcc
fix: cjs build of js-levenshtein
kettanaito Apr 29, 2023
229c967
chore(release): v0.0.0-fetch.rc-12
kettanaito Apr 29, 2023
cb565c9
fix: add "chalk@4" back to dependencies
kettanaito May 2, 2023
e20814e
chore(release): v0.0.0-fetch.rc-13
kettanaito May 2, 2023
b631430
Merge branch 'main' into feat/standard-api
kettanaito May 2, 2023
50c4c9c
fix: stream response chunks as they come in Node.js (#1606)
kettanaito May 3, 2023
8a527e9
fix: prevent "abort" listener memory leak in RequestHandler (#1608)
kettanaito May 7, 2023
b159c28
chore(release): v0.0.0-fetch.rc-14
kettanaito May 8, 2023
ed9275a
test: add response FormData browser test
kettanaito May 9, 2023
f663eeb
test: add Blob response browser test
kettanaito May 9, 2023
9c73b7d
test: add Blob json to request browser tests
kettanaito May 9, 2023
ccc576d
Merge branch 'main' into feat/standard-api
kettanaito May 12, 2023
2a287bc
feat: add "useRemoteHandlers" sync layer
kettanaito May 12, 2023
a20d446
fix: defer server connection to the first handler
kettanaito May 13, 2023
e9cc153
chore: use request/response serialize/deserialize utils
kettanaito May 13, 2023
7587800
chore: reuse req/res serialization for logging
kettanaito May 13, 2023
68727eb
chore: annotate sync server events map
kettanaito May 13, 2023
a2efc29
fix: bypass internal socket requests using "x-msw-request-type" header
kettanaito May 13, 2023
df07102
chore: use a single SYNC_SERVER_URL variable
kettanaito May 13, 2023
ceb507a
chore: use node polyfills for node tests in jest
kettanaito May 13, 2023
8bff5a0
feat: add "setupRemoteServer" API
kettanaito May 16, 2023
cedba0a
fix: await the server connection promise correctly
kettanaito May 17, 2023
598b1b7
fix: provide explicit exports from "setupRemoteServer"
kettanaito May 17, 2023
612a5ba
feat: support custom "MSW_INTERNAL_WEBSOCKET_PORT" env variable
kettanaito May 26, 2023
0f5c443
test(setupRemoteServer): add basic "use" test
kettanaito May 26, 2023
d27fc23
chore: move client creation to "setupRemoteServer"
kettanaito May 28, 2023
6e0c634
fix: format errors
kettanaito May 28, 2023
0bd22bb
feat(setupRemoteServer): support life-cycle events
kettanaito Jun 6, 2023
589eac6
Merge branch 'main' into feat/standard-api
kettanaito Jun 8, 2023
883d21f
Merge branch 'feat/standard-api' into feat/ws-sync-handlers
kettanaito Jun 8, 2023
b25655c
fix: normalize response status and status text when logging
kettanaito Jun 9, 2023
17f72e8
Merge branch 'main' into feat/standard-api
kettanaito Jun 9, 2023
f78d69e
Merge branch 'feat/standard-api' into feat/ws-sync-handlers
kettanaito Jun 13, 2023
a7caf12
test: add "Response.error" integration test
kettanaito Jun 29, 2023
82e3812
feat: do not set "x-powered-by" mocked response header
kettanaito Jun 29, 2023
8ca0287
chore: upgrade to @mswjs/[email protected]
kettanaito Jun 29, 2023
1f06019
fix: rely on "isMockedResponse" to emit response events in node
kettanaito Jun 29, 2023
90f3989
feat: handle Response.error special use-case
kettanaito Jun 30, 2023
cff199d
test(response-patching): rely on custom "x-source" response header
kettanaito Jun 30, 2023
510f8b8
fix: upgrade to [email protected], remove "process" workaround
kettanaito Jun 30, 2023
59c3e52
chore(release): v0.0.0-fetch.rc-15
kettanaito Jun 30, 2023
789327a
Merge branch 'main' into feat/standard-api
kettanaito Jul 20, 2023
47fd803
feat: make life-cycle event listeners have object argument
kettanaito Jul 20, 2023
61f6f3b
feat: remove the ".printHandlers()" method
kettanaito Jul 20, 2023
c371f03
feat: deprecate "rest" in favor of "http" (#1673)
kettanaito Jul 31, 2023
8943e1d
docs(passthrough): improve jsdoc description
kettanaito Jul 31, 2023
15db686
fix: remove unused utils modules
kettanaito Jul 31, 2023
90b854a
chore(release): v0.0.0-fetch.rc-16
kettanaito Aug 6, 2023
6adc652
Merge branch 'main' into feat/standard-api
kettanaito Aug 18, 2023
6239a0f
Merge branch 'main' into feat/standard-api
kettanaito Aug 18, 2023
bb02e57
test(graphql): add test on anonymous operation warning (#1693)
kettanaito Aug 18, 2023
5f3dd2a
Merge branch 'main' into feat/standard-api
kettanaito Aug 25, 2023
3b8f57e
fix(graphql): clone request before parsing (#1714)
kettanaito Aug 25, 2023
3fb29e5
chore(release): v0.0.0-fetch.rc-17
kettanaito Aug 25, 2023
223c17a
Merge branch 'main' into feat/standard-api
kettanaito Sep 2, 2023
a2b61ba
docs: add multipart-data common issue to migrating
kettanaito Sep 2, 2023
5034e48
Merge branch 'main' into feat/standard-api
kettanaito Sep 7, 2023
4ba43a5
chore: upgrade "headers-polyfill" to 3.2.3
kettanaito Sep 7, 2023
0898517
chore(release): v0.0.0-fetch.rc-18
kettanaito Sep 7, 2023
c5d03b0
Merge branch 'feat/standard-api' into feat/ws-sync-handlers
kettanaito Sep 7, 2023
b2b634c
fix: remove "NetworkError" in favor of "HttpResponse.error" (#1730)
kettanaito Sep 7, 2023
faa71c4
chore(release): v0.0.0-fetch.rc-19
kettanaito Sep 8, 2023
3eca8c7
fix(graphql): support request handler options (#1739)
kettanaito Sep 16, 2023
91989df
feat(worker): move exception handling from the worker (#1734)
kettanaito Sep 16, 2023
f1ff76a
Merge branch 'feat/standard-api' into feat/ws-sync-handlers
kettanaito Sep 16, 2023
0e8d362
chore: merge "main" into "feat/ws-sync-handlers"
kettanaito Oct 26, 2023
b8e9560
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Oct 30, 2023
2325fd5
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Nov 16, 2023
b3c4eda
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Jan 7, 2024
1b85677
chore: minor adjustments
kettanaito Jan 7, 2024
c9d23c9
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Jan 8, 2024
d9cc36c
chore(wip): continue
kettanaito Jan 13, 2024
e89afa5
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Feb 20, 2024
d23dd25
fix(wip): fix broken "use()" runtime handlers
kettanaito Feb 20, 2024
e79f5b2
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Sep 4, 2024
81969e3
fix: client sync server handshake
kettanaito Sep 5, 2024
46decd1
fix(RemoteRequestHandler): await response on `parse` phase
kettanaito Sep 5, 2024
661113d
test: add response type tests
kettanaito Sep 5, 2024
cb50d9a
fix: use default remote port number
kettanaito Sep 5, 2024
38b0e27
docs: add jsdoc to listen options
kettanaito Sep 5, 2024
6e2ee96
feat: add `remote.boundary()`
kettanaito Nov 29, 2024
dbb3cd6
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Nov 29, 2024
49357ed
chore(wip): debugging
kettanaito Dec 1, 2024
a6c6f6f
fix: fix ws passthrough, adjust remote tests
kettanaito Dec 4, 2024
0b2e2c0
test(wip): add `remote.boundary` tests
kettanaito Dec 4, 2024
0e48062
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Dec 5, 2024
2eba1ad
feat: support boundary and `contextId`
kettanaito Dec 5, 2024
11c466b
chore: adjust tests to headers changes
kettanaito Dec 5, 2024
7ad582c
chore(wip): wip
kettanaito Dec 10, 2024
fa8a197
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Dec 12, 2024
57ab678
fix(setupServer): skip forwarding internal websocket events
kettanaito Dec 12, 2024
c069b25
fix(RemoteRequestHandler): handle socket disconnects
kettanaito Dec 12, 2024
c2d5c13
fix(serializeUtils): preserve empty request/response bodies
kettanaito Dec 12, 2024
657d7b1
chore: last days of websocket server
kettanaito Dec 13, 2024
5264ec4
feat: use http for remote request handling
kettanaito Dec 13, 2024
e6fac6d
chore: remove `console.log` to fix linting
kettanaito Dec 18, 2024
9d04f0a
Merge branch 'main' into feat/ws-sync-handlers
kettanaito Dec 18, 2024
cfc120c
fix: use consistent `localhost` for remote server
kettanaito Dec 18, 2024
310a2c0
feat: use random internal server port
kettanaito Dec 18, 2024
28875f9
chore: rewrite `RemoteClient` to `node:http`
kettanaito Jan 4, 2025
69a26dc
feat: forward life-cycle events over http
kettanaito Jan 6, 2025
f6cb07c
test: pass correct ArrayBuffer
kettanaito Jan 8, 2025
c21f0bd
test: move the life-cycle event forwarding test
kettanaito Jan 8, 2025
3826cad
fix(setupServer): print a dev utils error on failed remote connection
kettanaito Jan 8, 2025
d7d09fb
chore: hopefully, improve node and core ts configs
kettanaito Jan 8, 2025
ac6da8b
test(setupServer): add test for not emitting life cycle events on int…
kettanaito Jan 8, 2025
b5ea6fc
fix: support `onUnhandledRequest`
kettanaito Jan 8, 2025
a70b518
chore: move `RemoteClient` to core
kettanaito Jan 8, 2025
0d03db2
test: add `onUnhandledRequest` default behavior test
kettanaito Jan 9, 2025
b22d00c
fix(node): export `ListenOptions`
kettanaito Jan 9, 2025
b0b0381
test: add warn/error/bypass `onUnhandledRequest` tests
kettanaito Jan 9, 2025
61c8fe7
test: add custom callback `onUnhandledRequest` test
kettanaito Jan 9, 2025
bc0e0a7
chore: move `spyOnLifeCycleEvents` to node utils
kettanaito Jan 9, 2025
3c76156
fix: set "duplex" to "half" when handling requests
kettanaito Jan 10, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
chore: move RemoteClient to core
  • Loading branch information
kettanaito committed Jan 8, 2025

Verified

This commit was signed with the committer’s verified signature.
rgoldberg Ross Goldberg
commit a70b5183750b8fb311c5b09481255f862d20c2a7
309 changes: 309 additions & 0 deletions src/core/RemoteClient.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,309 @@
import * as http from 'node:http'
import { Readable } from 'node:stream'
import { DeferredPromise } from '@open-draft/deferred-promise'
import { FetchResponse } from '@mswjs/interceptors'
import { invariant } from 'outvariant'
import type { LifeCycleEventsMap } from './sharedOptions'
import { bypass } from './bypass'
import { delay } from './delay'

export type ForwardedLifeCycleEventPayload = {
type: keyof LifeCycleEventsMap
args: {
requestId: string
request: {
method: string
url: string
headers: Array<[string, string]>
body: Uint8Array | null
}
response?: {
status: number
statusText: string
headers: Array<[string, string]>
body: Uint8Array | null
}
error?: {
name: string
message: string
stack?: string
}
}
}

export class RemoteClient {
public connected: boolean

protected agent: http.Agent

constructor(private readonly url: URL) {
this.agent = new http.Agent({
// Reuse the same socket between requests so we can communicate
// request's life-cycle events via HTTP more efficiently.
keepAlive: true,
})
this.connected = false
}

public async connect(): Promise<void> {
if (this.connected) {
return
}

const maxRetries = 4
let retries = 0

const tryConnect = (): Promise<void> => {
const connectionPromise = new DeferredPromise<void>()

const request = http
.request(this.url, {
agent: this.agent,
method: 'HEAD',
headers: {
accept: 'msw/passthrough',
},
timeout: 1000,
})
.end()

request
.once('response', (response) => {
if (response.statusCode === 200) {
connectionPromise.resolve()
} else {
connectionPromise.reject()
}
})
.once('error', () => {
connectionPromise.reject()
})
.once('timeout', () => {
connectionPromise.reject()
})

return connectionPromise.then(
() => {
this.connected = true
},
async () => {
invariant(
retries < maxRetries,
'Failed to connect to the remote server after %s retries',
maxRetries,
)

retries++
request.removeAllListeners()
return delay(500).then(() => tryConnect())
},
)
}

return tryConnect()
}

public async handleRequest(args: {
requestId: string
boundaryId: string
request: Request
}): Promise<Response | undefined> {
invariant(
this.connected,
'Failed to handle request "%s %s": client is not connected',
args.request.method,
args.request.url,
)

const fetchRequest = bypass(args.request, {
headers: {
accept: 'msw/internal',
'x-msw-request-url': args.request.url,
'x-msw-request-id': args.requestId,
'x-msw-boundary-id': args.boundaryId,
},
})
const request = http.request(this.url, {
method: fetchRequest.method,
headers: Object.fromEntries(fetchRequest.headers),
})

if (fetchRequest.body) {
Readable.fromWeb(fetchRequest.body as any).pipe(request, { end: true })
} else {
request.end()
}

const responsePromise = new DeferredPromise<Response | undefined>()

request
.once('response', (response) => {
if (response.statusCode === 404) {
responsePromise.resolve(undefined)
return
}

const fetchResponse = new FetchResponse(
/** @fixme Node.js types incompatibility */
Readable.toWeb(response) as ReadableStream<any>,
{
url: fetchRequest.url,
status: response.statusCode,
statusText: response.statusMessage,
headers: FetchResponse.parseRawHeaders(response.rawHeaders),
},
)
responsePromise.resolve(fetchResponse)
})
.once('error', () => {
responsePromise.resolve(undefined)
})
.once('timeout', () => {
responsePromise.resolve(undefined)
})

return responsePromise
}

public async handleLifeCycleEvent<
EventType extends keyof LifeCycleEventsMap,
>(event: {
type: EventType
args: LifeCycleEventsMap[EventType][0]
}): Promise<void> {
invariant(
this.connected,
'Failed to forward life-cycle events for "%s %s": remote client not connected',
event.args.request.method,
event.args.request.url,
)

const url = new URL('/life-cycle-events', this.url)
const payload = JSON.stringify({
type: event.type,
args: {
requestId: event.args.requestId,
request: await serializeFetchRequest(event.args.request),
response:
'response' in event.args
? await serializeFetchResponse(event.args.response)
: undefined,
error:
'error' in event.args ? serializeError(event.args.error) : undefined,
},
} satisfies ForwardedLifeCycleEventPayload)

invariant(
payload,
'Failed to serialize a life-cycle event "%s" for request "%s %s"',
event.type,
event.args.request.method,
event.args.request.url,
)

const donePromise = new DeferredPromise<void>()

http
.request(
url,
{
method: 'POST',
headers: {
accept: 'msw/passthrough, msw/internal',
'content-type': 'application/json',
},
},
(response) => {
if (response.statusCode === 200) {
donePromise.resolve()
} else {
donePromise.reject(
new Error(
`Failed to forward life-cycle event "${event.type}" for request "${event.args.request.method} ${event.args.request.url}": expected a 200 response but got ${response.statusCode}`,
),
)
}
},
)
.end(payload)
.once('error', (error) => {
// eslint-disable-next-line no-console
console.error(error)
donePromise.reject(
new Error(
`Failed to forward life-cycle event "${event.type}" for request "${event.args.request.method} ${event.args.request.url}": unexpected error. There's likely additional information above.`,
),
)
})

return donePromise
}
}

export async function serializeFetchRequest(
request: Request,
): Promise<ForwardedLifeCycleEventPayload['args']['request']> {
return {
url: request.url,
method: request.method,
headers: Array.from(request.headers),
body: request.body
? new Uint8Array(await request.clone().arrayBuffer())
: null,
}
}

export function deserializeFetchRequest(
value: NonNullable<ForwardedLifeCycleEventPayload['args']['request']>,
): Request {
return new Request(value.url, {
method: value.method,
headers: value.headers,
body: value.body,
})
}

async function serializeFetchResponse(
response: Response,
): Promise<ForwardedLifeCycleEventPayload['args']['response']> {
return {
status: response.status,
statusText: response.statusText,
headers: Array.from(response.headers),
body: response.body
? new Uint8Array(await response.clone().arrayBuffer())
: null,
}
}

export function deserializeFetchResponse(
value: NonNullable<ForwardedLifeCycleEventPayload['args']['response']>,
): Response {
return new FetchResponse(
value.body ? new Uint8Array(Object.values(value.body)) : null,
{
status: value.status,
statusText: value.statusText,
headers: value.headers,
},
)
}

export function serializeError(
error: Error,
): ForwardedLifeCycleEventPayload['args']['error'] {
return {
name: error.name,
message: error.message,
stack: error.stack,
}
}

export function deserializeError(
value: NonNullable<ForwardedLifeCycleEventPayload['args']['error']>,
): Error {
const error = new Error(value.message)
error.name = value.name
error.stack = value.stack
return error
}
2 changes: 1 addition & 1 deletion src/core/handlers/RemoteRequestHandler.ts
Original file line number Diff line number Diff line change
@@ -5,7 +5,7 @@ import {
type ResponseResolver,
type RequestHandlerDefaultInfo,
} from './RequestHandler'
import { RemoteClient } from '../../node/setupRemoteServer'
import type { RemoteClient } from '../RemoteClient'

interface RemoteRequestHandlerParsedResult {
response: Response | undefined
2 changes: 1 addition & 1 deletion src/node/SetupServerApi.ts
Original file line number Diff line number Diff line change
@@ -8,12 +8,12 @@ import type { RequestHandler } from '~/core/handlers/RequestHandler'
import type { ListenOptions, SetupServer } from './glossary'
import type { WebSocketHandler } from '~/core/handlers/WebSocketHandler'
import { SetupServerCommonApi } from './SetupServerCommonApi'
import { RemoteClient } from './setupRemoteServer'
import { RemoteRequestHandler } from '~/core/handlers/RemoteRequestHandler'
import { shouldBypassRequest } from '~/core/utils/internal/requestUtils'
import { getRemoteContextFromEnvironment } from './remoteContext'
import { LifeCycleEventsMap } from '~/core/sharedOptions'
import { devUtils } from '~/core/utils/internal/devUtils'
import { RemoteClient } from '~/core/RemoteClient'

const handlersStorage = new AsyncLocalStorage<RequestHandlersContext>()

Loading
Loading