Skip to content

Commit

Permalink
Added support for transport overrides (#253)
Browse files Browse the repository at this point in the history
* add support for transport overrides

* lint

* change transport override keyword
  • Loading branch information
karen-stepanyan authored Apr 11, 2024
1 parent 3294f89 commit 4cf52e0
Show file tree
Hide file tree
Showing 4 changed files with 194 additions and 26 deletions.
10 changes: 8 additions & 2 deletions src/adapter/endpoint.ts
Original file line number Diff line number Diff line change
Expand Up @@ -170,13 +170,19 @@ export class AdapterEndpoint<T extends EndpointGenerics> implements AdapterEndpo
}

/**
* Default routing strategy. Will try and use the transport input parameter if present in the request body.
* Default routing strategy. Will try and use the transport override if present
* or transport input parameter in the request body.
*
* @param req - The current adapter request
* @returns the transport param if present
* @returns the transport param or override if present
*/
private defaultRouter(req: AdapterRequest<TypeFromDefinition<T['Parameters']>>) {
const rawRequestBody = req.body as unknown as { data: AdapterRequestData }
const requestOverrides = rawRequestBody.data?.overrides?.[this.adapterName.toLowerCase()]
// Transport override
if (requestOverrides?.['transport']) {
return requestOverrides['transport']
}
return rawRequestBody.data?.transport
}
}
5 changes: 3 additions & 2 deletions src/util/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ export type AdapterRequestContext<T> = {
/** Name of the endpoint this payload should be directed to */
endpointName: string

/** Name of the endpoint this payload should be directed to */
/** Name of the transport this payload should be directed to */
transportName: string

/** Precalculated cache key used to get and set corresponding values from the cache and subscription sets */
Expand Down Expand Up @@ -101,11 +101,12 @@ export interface AdapterMetricsMeta {
}

/**
* Map of overrides objects (symbol -\> symbol) per adapter name
* Map of overrides objects (symbol -\> symbol and request transport) per adapter name
*/
export type Overrides = {
[adapterName: string]: {
[symbol: string]: string
transport: string
}
}

Expand Down
2 changes: 1 addition & 1 deletion src/validation/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,7 @@ export const validatorMiddleware: AdapterMiddlewareBuilder =

req.requestContext.cacheKey = `${cachePrefix}${cacheKey}`
} else {
const transportName = endpoint.getTransportNameForRequest(req, adapter.config.settings)
const transportName = req.requestContext.transportName
req.requestContext.cacheKey = calculateCacheKey({
data: req.requestContext.data,
adapterName: adapter.name,
Expand Down
203 changes: 182 additions & 21 deletions test/transports/routing.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import untypedTest, { TestFn } from 'ava'
import axios, { AxiosRequestConfig, AxiosResponse } from 'axios'
import MockAdapter from 'axios-mock-adapter'
import { Server } from 'mock-socket'
import { Adapter, AdapterEndpoint, EndpointContext } from '../../src/adapter'
import { Adapter, AdapterEndpoint } from '../../src/adapter'
import {
AdapterConfig,
SettingsDefinitionFromConfig,
Expand All @@ -18,6 +18,8 @@ import {
} from '../../src/transports'
import { InputParameters } from '../../src/validation'
import { TestAdapter, mockWebSocketProvider } from '../../src/util/testing-utils'
import { AdapterRequest } from '../../src/util'
import { TypeFromDefinition } from '../../src/validation/input-params'

const test = untypedTest as TestFn<{
testAdapter: TestAdapter<SettingsDefinitionFromConfig<typeof adapterConfig>>
Expand Down Expand Up @@ -88,7 +90,7 @@ const inputParameters = new InputParameters({
})

class MockWebSocketTransport extends WebSocketTransport<WebSocketTypes> {
public backgroundExecuteCalls = 0
public registerRequestCalls = 0

constructor() {
super({
Expand Down Expand Up @@ -123,9 +125,12 @@ class MockWebSocketTransport extends WebSocketTransport<WebSocketTypes> {
})
}

override async backgroundExecute(context: EndpointContext<WebSocketTypes>): Promise<void> {
this.backgroundExecuteCalls++
return super.backgroundExecute(context)
override async registerRequest(
req: AdapterRequest<TypeFromDefinition<WebSocketTypes['Parameters']>>,
_: WebSocketTypes['Settings'],
): Promise<void> {
this.registerRequestCalls++
return super.registerRequest(req, _)
}
}

Expand Down Expand Up @@ -157,9 +162,11 @@ type HttpTypes = BaseEndpointTypes & {
}

class MockHttpTransport extends HttpTransport<HttpTypes> {
backgroundExecuteCalls = 0
// Since backgroundExecute always runs for all compatible transports, regardless of the requests,
// we check for registered requests
registerRequestCalls = 0

constructor(private callSuper = false) {
constructor() {
super({
prepareRequests: (params) => {
return {
Expand All @@ -186,11 +193,12 @@ class MockHttpTransport extends HttpTransport<HttpTypes> {
})
}

override async backgroundExecute(context: EndpointContext<HttpTypes>): Promise<void> {
this.backgroundExecuteCalls++
if (this.callSuper) {
super.backgroundExecute(context)
}
override async registerRequest(
req: AdapterRequest<TypeFromDefinition<HttpTypes['Parameters']>>,
_: HttpTypes['Settings'],
): Promise<void> {
this.registerRequestCalls++
return super.registerRequest(req, _)
}
}

Expand All @@ -204,7 +212,7 @@ type SSETypes = BaseEndpointTypes & {
}

class MockSseTransport extends SseTransport<SSETypes> {
public backgroundExecuteCalls = 0
public registerRequestCalls = 0

constructor() {
super({
Expand Down Expand Up @@ -251,9 +259,12 @@ class MockSseTransport extends SseTransport<SSETypes> {
})
}

override async backgroundExecute(context: EndpointContext<SSETypes>): Promise<void> {
this.backgroundExecuteCalls++
return super.backgroundExecute(context)
override async registerRequest(
req: AdapterRequest<TypeFromDefinition<SSETypes['Parameters']>>,
_: SSETypes['Settings'],
): Promise<void> {
this.registerRequestCalls++
return super.registerRequest(req, _)
}
}

Expand Down Expand Up @@ -299,6 +310,15 @@ test.beforeEach(async (t) => {
}
})

test.afterEach(() => {
const batchTransport = transports.get('batch') as unknown as { registerRequestCalls: number }
const wsTransport = transports.get('websocket') as unknown as { registerRequestCalls: number }
const sseTransport = transports.get('sse') as unknown as { registerRequestCalls: number }
batchTransport.registerRequestCalls = 0
wsTransport.registerRequestCalls = 0
sseTransport.registerRequestCalls = 0
})

test.serial('endpoint routing errors on invalid transport', async (t) => {
t.is(
Object.keys(transports).find((s) => s === 'INVALID'),
Expand Down Expand Up @@ -340,7 +360,7 @@ test.serial('endpoint routing can route to HttpTransport', async (t) => {

t.is(error.statusCode, 504)
const internalTransport = transports.get('batch') as unknown as MockHttpTransport
t.assert(internalTransport.backgroundExecuteCalls > 0)
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('endpoint routing can route to WebSocket transport', async (t) => {
Expand All @@ -351,7 +371,7 @@ test.serial('endpoint routing can route to WebSocket transport', async (t) => {
})
t.is(error?.statusCode, 504)
const internalTransport = transports.get('websocket') as unknown as MockWebSocketTransport
t.assert(internalTransport.backgroundExecuteCalls > 0)
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('endpoint routing can route to SSE transport', async (t) => {
Expand All @@ -377,7 +397,7 @@ test.serial('endpoint routing can route to SSE transport', async (t) => {
t.is(error.statusCode, 504)

const internalTransport = transports.get('sse') as unknown as MockSseTransport
t.assert(internalTransport.backgroundExecuteCalls > 0)
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('custom router is applied to get valid transport to route to', async (t) => {
Expand Down Expand Up @@ -439,7 +459,7 @@ test.serial('custom router is applied to get valid transport to route to', async
t.is(error.statusCode, 504)

const internalTransport = transports.get('batch') as unknown as MockHttpTransport
t.assert(internalTransport.backgroundExecuteCalls > 0)
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('custom router returns invalid transport and request fails', async (t) => {
Expand Down Expand Up @@ -624,7 +644,7 @@ test.serial('missing transport in input params with default succeeds', async (t)
t.is(error.statusCode, 504)

const internalTransport = transports.get('batch') as unknown as MockHttpTransport
t.assert(internalTransport.backgroundExecuteCalls > 0)
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('transport creation fails if transport names are not acceptable', async (t) => {
Expand Down Expand Up @@ -658,3 +678,144 @@ test.serial('transports with same name throws error', async (t) => {
{ message: 'Transport with name "websocket" is already registered in this map' },
)
})

test.serial('transport override routes to correct Transport', async (t) => {
axiosMock
.onPost(`${restUrl}/price`, {
pairs: [
{
base: from,
quote: to,
},
],
})
.reply(200, {
prices: [
{
pair: `${from}/${to}`,
price,
},
],
})

const error = await t.context.testAdapter.request({
from,
to,
transport: 'websocket',
overrides: {
test: {
transport: 'batch',
},
},
})

t.is(error.statusCode, 504)
const internalTransport = transports.get('batch') as unknown as MockHttpTransport
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial('invalid transport override is skipped', async (t) => {
axiosMock
.onPost(`${restUrl}/price`, {
pairs: [
{
base: from,
quote: to,
},
],
})
.reply(200, {
prices: [
{
pair: `${from}/${to}`,
price,
},
],
})

const error = await t.context.testAdapter.request({
from,
to,
transport: 'websocket',
overrides: {
// Invalid adapter name
XXXX: {
transport: 'batch',
},
},
})

t.is(error.statusCode, 504)
const internalTransport = transports.get('websocket') as unknown as MockHttpTransport
t.assert(internalTransport.registerRequestCalls > 0)
})

test.serial(
'transport and transport override are ignored when custom router returns a value',
async (t) => {
const endpoint = new AdapterEndpoint<BaseEndpointTypes>({
inputParameters,
name: 'price', // /price
transportRoutes: transports,
customRouter: () => 'batch',
})

const customConfig = new AdapterConfig(settings, {
envDefaultOverrides: {
LOG_LEVEL: 'debug',
METRICS_ENABLED: false,
CACHE_POLLING_SLEEP_MS: 10,
CACHE_POLLING_MAX_RETRIES: 0,
},
})

const adapter = new Adapter({
name: 'TEST',
defaultEndpoint: 'price',
config: customConfig,
endpoints: [endpoint],
rateLimiting: {
tiers: {
default: {
rateLimit1s: 5,
},
},
},
})

const testAdapter = await TestAdapter.start(adapter, t.context)

axiosMock
.onPost(`${restUrl}/price`, {
pairs: [
{
base: from,
quote: to,
},
],
})
.reply(200, {
prices: [
{
pair: `${from}/${to}`,
price,
},
],
})

const error = await testAdapter.request({
from,
to,
transport: 'sse',
overrides: {
test: {
transport: 'websocket',
},
},
})
t.is(error.statusCode, 504)

const internalTransport = transports.get('batch') as unknown as MockHttpTransport
t.assert(internalTransport.registerRequestCalls > 0)
},
)

0 comments on commit 4cf52e0

Please sign in to comment.