Configurabe, light weight, integration testing context that can be used to support writing integrations with the test runner of your choice.
export const configureIntegrationTestCtxProvider = <
ENVKEYS extends string = string,
MOCKSERVERNAMES extends string = string,
WHENDELTA extends object = object,
APICLIENT extends object = RestClient,
>(
clientAndServerProvider: ClientAndServerProvider<ENVKEYS, APICLIENT>,
logger: TestamaticLogger,
envConfig: EnvConfig<ENVKEYS> = nullEnvConfig,
whenDeltaConfig?: WhenDeltaConfig<WHENDELTA>,
mockHttpServers: MockHttpServer<MOCKSERVERNAMES, ENVKEYS>[] = [],
beforeAll: Given[] = [],
beforeEach: Given[] = [],
): IntegrationTestCtxProvider<ENVKEYS, MOCKSERVERNAMES, WHENDELTA, APICLIENT>
export type IntegrationTestCtx<
ENVKEYS extends string,
MOCKSERVERNAMES extends string,
WHENDELTA extends object,
APICLIENT extends object,
> = {
each: BeforeAndAfter
all: BeforeAndAfter
httpMock: MockServerExpecter<MOCKSERVERNAMES, ENVKEYS>
when: WHEN<WHENDELTA>
api: API<APICLIENT>
env: EnvVars<ENVKEYS>
}
There are a few key types that must be provided before configuring the test context for a particular project
A string type representing all the env keys that the system relies on.
export type MyEnvKeys = "Key1" | "Key2" | "Key3"
Names fo all the mock servers that are to be created.
export type MyMockServerNames = "jwksServer" | "oauthServer" | "someOtherServer"
Any object structure that will be calculated and returned every time the test context when is called
export type MyDelta = { eventSourceTableRows: { event: object } }
Simple interface used to create the app and a corresponding client to be used to call the app
export type ClientAndServerProvider<ENVKEYS extends string, CLIENT extends object = RestClient> = (
env: EnvVars<ENVKEYS>,
) => Promise<ClientAndServer<CLIENT>>
export type ClientAndServer<APICLIENT extends object = RestClient> = { client: APICLIENT; close: () => Promise<void> }
Simple interface used to log output from testamatic
export interface TestamaticLogger {
fatal(message: string, data?: object): void
error(message: string, error: unknown): void
warn(message: string, data?: object): void
info(message: string, data?: object): void
debug(message: string, data?: object): void
trace(message: string, data?: object): void
child(name: string, data?: object): TestamaticLogger
}
Simple interface that creates the environment in the correct location ie local env, parameterstore .... This interface is called once all the mocks have been created and thier urls are known.
export type EnvSetup<ENVKEYS extends string> = {
setup: (env: EnvVars<ENVKEYS>) => Promise<void>
teardown: () => Promise<void>
}
The when method provided by the test context can return a delta, which is the change of anything of interest. For example the change in a table might be of interesting to assert against. Once configured this snaphot is calculated every time when is called and returned along with the response from what was executed.
when: <RES>(isExecuted: IsExecuted<RES>) => Promise<{ response: RES; delta: WHENDELTA }>
The following interface controls how a snapshot is taken before and after running a 'when' and how the delta between them is calculated.
export type WhenDeltaConfig<WHENDELTA> = {
snapshot: () => Promise<WHENDELTA>
diff: (first: WHENDELTA, second: WHENDELTA) => Promise<WHENDELTA>
}
Simple servers that can be configured to expect input and return responses in a similar way that we can use mocks in unit testing except http mocks test everything because they actually receive TCP requests and return TCP responses. Expected behaviours are set up via the httpMock.expect method supplied by the test context.
httpMock: {
expect: (name: MOCKSERVERNAMES, expectation: MockHttpExpectation) => void
}
A list of http mocks can be provided in the makeIntegrationTestContextFactory configuration call above.
export class MockHttpServer<MOCKSERVERNAMES extends string, ENVKEYS extends string> {
constructor(
public name: MOCKSERVERNAMES,
private urlEnvKey: ENVKEYS,
host: string = "localhost",
private protocol: "http" | "https" = "http",
private listenerFactory: MockTcpListenerFactory
) {
}
}
Setups that are called once at the beginning of a test run, and once at the end to teardown
export interface Given {
teardown(): Promise<void>
setup(): Promise<void>
}
Setups that are called twice before each test. Once to teardown and then to setup. This means that state remains once a test is run so it can be used to debug or trouble shoot.
export interface Given {
teardown(): Promise<void>
setup(): Promise<void>
}
Are found in integrationtesting package in the form of running integration tests.
Curently calls clientAndServerProvider once at the beginning of the test run. May be a need to call create every time to avoid issues around caching etc
- .devcontainer -> all docker related config
- src -> all typescript src code
- Docker
- Bash
- npm
Now you can run the rest of the commands.
Any npm script prefixed with 'docker:' is designed to be run on the host machine
nom run docker:dev:connect
Any npm script not prefixed with 'docker:' is designed to be run inside a container
- compile -> code compilation, t need to get inside a container by running:
npm run compile
npm run test:micro
npm run docker:dev:connect
npm i
npm run build
npm run test:micro
npm run test:integration
A useful way to clobber all docker containers running on your machine. Unlock docker:down it can be run from anywhere and it will clobber all running containers regardless of what created them.