grafana/public/test/core/redux/reducerTester.ts
Esteban Beltran 0b03344baa
Chore: Update to typescript 5 (#76511)
* Update dependency typescript to v5

* Update yarn.lock

* Fix typescript errors

* Update typescript version sdk

* Revert useDescription.ts

* Fix ts errors

* Fix Typescript errors after Symbol.unscopables type change

* Fix colormanipulator errors

* Update packages/grafana-data/src/vector/FunctionalVector.ts

* Fix ts errors in dashboardmigrator

* Fix sandbox component typescript error

* Update yarn

* Update to typescript 5.2

* Fix typescript error

* update typescript/vscode patch/sdk/whatever

* fix ts errors in elasticsearch

* Fix two errors in alerting

* Fix error in dashboard-scene

* Fix errors in dashboard tests

* Fix errors in explore tests

* Fix error in plugins sandbox

* fix error in DashboardQueryRunner

* fix errors in grafana-data

* fix errors in PanelChrome story

* update betterer

* better fix for cloud monitoring

* fix error in reducer tester

---------

Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
Co-authored-by: Zoltán Bedi <zoltan.bedi@gmail.com>
Co-authored-by: Josh Hunt <joshhunt@users.noreply.github.com>
Co-authored-by: joshhunt <josh@trtr.co>
2023-11-07 13:20:20 +00:00

110 lines
2.9 KiB
TypeScript

import { AnyAction } from '@reduxjs/toolkit';
import { cloneDeep } from 'lodash';
import { Action } from 'redux';
import { StoreState } from 'app/types';
type GrafanaReducer<S = StoreState, A extends Action = AnyAction> = (state: S, action: A) => S;
export interface Given<State> {
givenReducer: (
reducer: GrafanaReducer<State, AnyAction>,
state: State,
showDebugOutput?: boolean,
disableDeepFreeze?: boolean
) => When<State>;
}
export interface When<State> {
whenActionIsDispatched: (action: AnyAction) => Then<State>;
}
export interface Then<State> {
thenStateShouldEqual: (state: State) => When<State>;
thenStatePredicateShouldEqual: (predicate: (resultingState: State) => boolean) => When<State>;
whenActionIsDispatched: (action: AnyAction) => Then<State>;
}
const isNotException = (object: unknown, propertyName: string) =>
typeof object === 'function'
? propertyName !== 'caller' && propertyName !== 'callee' && propertyName !== 'arguments'
: true;
export const deepFreeze = <T>(obj: T): T => {
if (typeof obj === 'object') {
for (const key in obj) {
const prop = obj[key];
if (
prop &&
Object.hasOwn(obj, key) &&
isNotException(obj, key) &&
(typeof prop === 'object' || typeof prop === 'function') &&
!Object.isFrozen(prop)
) {
deepFreeze(prop);
}
}
}
return Object.freeze(obj);
};
interface ReducerTester<State> extends Given<State>, When<State>, Then<State> {}
export const reducerTester = <State>(): Given<State> => {
let reducerUnderTest: GrafanaReducer<State, AnyAction>;
let resultingState: State;
let initialState: State;
let showDebugOutput = false;
const givenReducer = (
reducer: GrafanaReducer<State, AnyAction>,
state: State,
debug = false,
disableDeepFreeze = false
): When<State> => {
reducerUnderTest = reducer;
initialState = cloneDeep(state);
if (!disableDeepFreeze && (typeof state === 'object' || typeof state === 'function')) {
deepFreeze(initialState);
}
showDebugOutput = debug;
return instance;
};
const whenActionIsDispatched = (action: AnyAction): Then<State> => {
resultingState = reducerUnderTest(resultingState || initialState, action);
return instance;
};
const thenStateShouldEqual = (state: State): When<State> => {
if (showDebugOutput) {
console.log(JSON.stringify(resultingState, null, 2));
}
expect(resultingState).toEqual(state);
return instance;
};
const thenStatePredicateShouldEqual = (predicate: (resultingState: State) => boolean): When<State> => {
if (showDebugOutput) {
console.log(JSON.stringify(resultingState, null, 2));
}
expect(predicate(resultingState)).toBe(true);
return instance;
};
const instance: ReducerTester<State> = {
thenStateShouldEqual,
thenStatePredicateShouldEqual,
givenReducer,
whenActionIsDispatched,
};
return instance;
};