213 lines
5.9 KiB
JavaScript
213 lines
5.9 KiB
JavaScript
'use strict'
|
|
|
|
const { kClients } = require('../core/symbols')
|
|
const Agent = require('../dispatcher/agent')
|
|
const {
|
|
kAgent,
|
|
kMockAgentSet,
|
|
kMockAgentGet,
|
|
kDispatches,
|
|
kIsMockActive,
|
|
kNetConnect,
|
|
kGetNetConnect,
|
|
kOptions,
|
|
kFactory,
|
|
kMockAgentRegisterCallHistory,
|
|
kMockAgentIsCallHistoryEnabled,
|
|
kMockAgentAddCallHistoryLog,
|
|
kMockAgentMockCallHistoryInstance,
|
|
kMockCallHistoryAddLog
|
|
} = require('./mock-symbols')
|
|
const MockClient = require('./mock-client')
|
|
const MockPool = require('./mock-pool')
|
|
const { matchValue, buildAndValidateMockOptions } = require('./mock-utils')
|
|
const { InvalidArgumentError, UndiciError } = require('../core/errors')
|
|
const Dispatcher = require('../dispatcher/dispatcher')
|
|
const PendingInterceptorsFormatter = require('./pending-interceptors-formatter')
|
|
const { MockCallHistory } = require('./mock-call-history')
|
|
|
|
class MockAgent extends Dispatcher {
|
|
constructor (opts) {
|
|
super(opts)
|
|
|
|
const mockOptions = buildAndValidateMockOptions(opts)
|
|
|
|
this[kNetConnect] = true
|
|
this[kIsMockActive] = true
|
|
this[kMockAgentIsCallHistoryEnabled] = mockOptions?.enableCallHistory ?? false
|
|
|
|
// Instantiate Agent and encapsulate
|
|
if (opts?.agent && typeof opts.agent.dispatch !== 'function') {
|
|
throw new InvalidArgumentError('Argument opts.agent must implement Agent')
|
|
}
|
|
const agent = opts?.agent ? opts.agent : new Agent(opts)
|
|
this[kAgent] = agent
|
|
|
|
this[kClients] = agent[kClients]
|
|
this[kOptions] = mockOptions
|
|
|
|
if (this[kMockAgentIsCallHistoryEnabled]) {
|
|
this[kMockAgentRegisterCallHistory]()
|
|
}
|
|
}
|
|
|
|
get (origin) {
|
|
let dispatcher = this[kMockAgentGet](origin)
|
|
|
|
if (!dispatcher) {
|
|
dispatcher = this[kFactory](origin)
|
|
this[kMockAgentSet](origin, dispatcher)
|
|
}
|
|
return dispatcher
|
|
}
|
|
|
|
dispatch (opts, handler) {
|
|
// Call MockAgent.get to perform additional setup before dispatching as normal
|
|
this.get(opts.origin)
|
|
|
|
this[kMockAgentAddCallHistoryLog](opts)
|
|
|
|
return this[kAgent].dispatch(opts, handler)
|
|
}
|
|
|
|
async close () {
|
|
this.clearCallHistory()
|
|
await this[kAgent].close()
|
|
this[kClients].clear()
|
|
}
|
|
|
|
deactivate () {
|
|
this[kIsMockActive] = false
|
|
}
|
|
|
|
activate () {
|
|
this[kIsMockActive] = true
|
|
}
|
|
|
|
enableNetConnect (matcher) {
|
|
if (typeof matcher === 'string' || typeof matcher === 'function' || matcher instanceof RegExp) {
|
|
if (Array.isArray(this[kNetConnect])) {
|
|
this[kNetConnect].push(matcher)
|
|
} else {
|
|
this[kNetConnect] = [matcher]
|
|
}
|
|
} else if (typeof matcher === 'undefined') {
|
|
this[kNetConnect] = true
|
|
} else {
|
|
throw new InvalidArgumentError('Unsupported matcher. Must be one of String|Function|RegExp.')
|
|
}
|
|
}
|
|
|
|
disableNetConnect () {
|
|
this[kNetConnect] = false
|
|
}
|
|
|
|
enableCallHistory () {
|
|
this[kMockAgentIsCallHistoryEnabled] = true
|
|
|
|
return this
|
|
}
|
|
|
|
disableCallHistory () {
|
|
this[kMockAgentIsCallHistoryEnabled] = false
|
|
|
|
return this
|
|
}
|
|
|
|
getCallHistory () {
|
|
return this[kMockAgentMockCallHistoryInstance]
|
|
}
|
|
|
|
clearCallHistory () {
|
|
if (this[kMockAgentMockCallHistoryInstance] !== undefined) {
|
|
this[kMockAgentMockCallHistoryInstance].clear()
|
|
}
|
|
}
|
|
|
|
// This is required to bypass issues caused by using global symbols - see:
|
|
// https://github.com/nodejs/undici/issues/1447
|
|
get isMockActive () {
|
|
return this[kIsMockActive]
|
|
}
|
|
|
|
[kMockAgentRegisterCallHistory] () {
|
|
if (this[kMockAgentMockCallHistoryInstance] === undefined) {
|
|
this[kMockAgentMockCallHistoryInstance] = new MockCallHistory()
|
|
}
|
|
}
|
|
|
|
[kMockAgentAddCallHistoryLog] (opts) {
|
|
if (this[kMockAgentIsCallHistoryEnabled]) {
|
|
// additional setup when enableCallHistory class method is used after mockAgent instantiation
|
|
this[kMockAgentRegisterCallHistory]()
|
|
|
|
// add call history log on every call (intercepted or not)
|
|
this[kMockAgentMockCallHistoryInstance][kMockCallHistoryAddLog](opts)
|
|
}
|
|
}
|
|
|
|
[kMockAgentSet] (origin, dispatcher) {
|
|
this[kClients].set(origin, dispatcher)
|
|
}
|
|
|
|
[kFactory] (origin) {
|
|
const mockOptions = Object.assign({ agent: this }, this[kOptions])
|
|
return this[kOptions] && this[kOptions].connections === 1
|
|
? new MockClient(origin, mockOptions)
|
|
: new MockPool(origin, mockOptions)
|
|
}
|
|
|
|
[kMockAgentGet] (origin) {
|
|
// First check if we can immediately find it
|
|
const client = this[kClients].get(origin)
|
|
if (client) {
|
|
return client
|
|
}
|
|
|
|
// If the origin is not a string create a dummy parent pool and return to user
|
|
if (typeof origin !== 'string') {
|
|
const dispatcher = this[kFactory]('http://localhost:9999')
|
|
this[kMockAgentSet](origin, dispatcher)
|
|
return dispatcher
|
|
}
|
|
|
|
// If we match, create a pool and assign the same dispatches
|
|
for (const [keyMatcher, nonExplicitDispatcher] of Array.from(this[kClients])) {
|
|
if (nonExplicitDispatcher && typeof keyMatcher !== 'string' && matchValue(keyMatcher, origin)) {
|
|
const dispatcher = this[kFactory](origin)
|
|
this[kMockAgentSet](origin, dispatcher)
|
|
dispatcher[kDispatches] = nonExplicitDispatcher[kDispatches]
|
|
return dispatcher
|
|
}
|
|
}
|
|
}
|
|
|
|
[kGetNetConnect] () {
|
|
return this[kNetConnect]
|
|
}
|
|
|
|
pendingInterceptors () {
|
|
const mockAgentClients = this[kClients]
|
|
|
|
return Array.from(mockAgentClients.entries())
|
|
.flatMap(([origin, scope]) => scope[kDispatches].map(dispatch => ({ ...dispatch, origin })))
|
|
.filter(({ pending }) => pending)
|
|
}
|
|
|
|
assertNoPendingInterceptors ({ pendingInterceptorsFormatter = new PendingInterceptorsFormatter() } = {}) {
|
|
const pending = this.pendingInterceptors()
|
|
|
|
if (pending.length === 0) {
|
|
return
|
|
}
|
|
|
|
throw new UndiciError(
|
|
pending.length === 1
|
|
? `1 interceptor is pending:\n\n${pendingInterceptorsFormatter.format(pending)}`.trim()
|
|
: `${pending.length} interceptors are pending:\n\n${pendingInterceptorsFormatter.format(pending)}`.trim()
|
|
)
|
|
}
|
|
}
|
|
|
|
module.exports = MockAgent
|