redux-devtools/packages/redux-devtools-rtk-query-monitor/src/selectors.ts
Fabrizio Vitale 7d92a5e186
[feat][monitor] Add rkt-query-inspector-monitor - feat/rtk query monitor (#750)
* chore: copy rtk-query example from toolkit

* feat(rtk-query): complete initial setup of rtk-query

* feat: complete inspector layout and add initial JSONTree setup

* fix: unintentional removal of tsconfig

* feat(search): add search logic and refactor monitor state shape

* fix: inverted monitor theme inside  devtoop-app

Othetr changes:

* simplify monitor integration

* fix: rtk monitor reducer not working in  app

* refactor(rtk-query): simplify theme config

* feat(rtk-query-monitor): add query preview tabs

* fix: wip

* refactor(examples): add rtk-query-polling to workspace

Other changes:

* docs(rtk-query-polling): add README.md

* chore(rtk-query-inspector): add demo to monorepo

Other changes:

chore: increase isWideScreen polling interval to 300

refactor: add subscription as root node in QueryPreviewSubscriptions

* feat(rtk-query): add multiple filter options

* chore(rtk-queery): rename demo build script and add SKIP_PREFLIGHT_CHECK=true

* feat(rtk-query): display status flags in QueryPreviewInfo

* chore(rtk-query): update typescript versions in rkt-inspector-monitor & its demo

* docs(rtk-query): add proper README

Other changes:

* fix examples/rtk-query-poilling

* docs(rtk-query): improve rtk-query-inspector-monitor demo gif

* docs(rtk-query): clean up demo

* fix(rtk-query): clear button not updating redux state

* docs(rtk-query): add link to rtk-query-inspector-monitor demo site

* chore(rtk-query): run prettier after prettier upgrade (55e2284)

* docs(rtk.query): clean up readme add features, todo and development section

* docs(rtk-query): fix link href

* chore(rtk-query): clean up rtk-query-imspector-monitor-demo and add post example

* feat(rtk-query): add counters on tags & subs tab buttons

* fix(rtk-query): layering issue between queryPreview tabList and select

Other changes:

* clean up demo styles

* run prettier

* fix: revert accidental changes packages/redux-devtools-serialize/tsconfig.json

* chore: change QueryComparators.fulfilledTimeStamp label

* feat(rtk-query): display api stats

* refactor: remove rtk-query-polling example from monorepo

* chore: regenerate lock file and add @types/react as monorepo devDep

* chore: display apiState

Other changes:

* fix close button overflow

* minor responsive style tweaks

* display reducerPath in query tab

* fix(rtk-query): resolve CI errors

- fix(prettier:check): unformatted file
- fix(lint:all): fix accidentallly removed .eslintignore

* chore(rtk-query): rename package to '@redux-devtools/rtk-query-monitor'

* fix(rtk-query): lint:all error

https://github.com/reduxjs/redux-devtools/runs/2869729048?check_suite_focus=true

* feat(rtk-query): add fallback message if there are no rtk-query apis

* refactor(rtk-query): move Inspector & Monitor to containers clean up typings

Other changes:

* chore: improved type coverage

* chore: do not lint packages/redux-devtools-rtk-query-monitor/demo folder

* refactor: put sort order buttons inside a component

* chore: hopefully resolve mockServiceWorker formatting issues

* fix(rtk-query): incorrect link color

Other changes:

* fix: add missing anchor property value noopener

* refactor(rtk-query): simplify sort order control

* feat(rtk-query): add timings to apiStats sections

* feat(rtk-query): add slowest and fastest in timings section

* feat(rtk-query): improve formatting of timings and display average loading time

* fix(rtk-query): rtk-query imports

* refactor(rtk-query): reduce selector computations

Other changes:

* simplify TreeView props

* feat(rtk-query): add actions tab

* refactor(rtk-query): add custom logic for TreeView shouldExpandNode

Other changes:

* feat: added duration in QueryPreviewInfo tab

* refactor: TreeView component

* chore(rtk-query): improve demo visibility on small devices

* feat(rtk-query): do not display tree node preview

Other changes:

* improve visibility of demo devTools on small devices

* tweak QueryPreviewInfo labels

* chore(rtk-query): improve responsiveness

* refactor(rtk-query): move preview to containers remove unnecessary computation

* feat(rtk-query): display median of timings

Other changes:

* improved shouldExpandNode logic of QueryPreviewActions

* tweaked mean logic

* refactor(rtk-query-monitor): conform demo setup to repo standards

* chore(rtk-query-monitor): add option to select active devtools

* chore(rtk-query-monitor): remove redux-devtools/examples/rtk-query-polling

* refactor(rtk-query): improve UI of api tab

* feat(rtk-query): add regex search

* feat(rtk-query): display mutations in queryList

* refactor(rtk-query): track all fulfilled requests using actions

Other changes:

* refactor(rtk-query): rename tally properties

* chore(rtk-query): update @redux-devtools/rtk-query-monitor dependencies

* fix(rtk-query): demo build failing caused by a typing error
2021-08-26 15:33:06 -04:00

284 lines
7.9 KiB
TypeScript

import { Action, createSelector, Selector } from '@reduxjs/toolkit';
import { RtkQueryInspectorProps } from './containers/RtkQueryInspector';
import {
ApiStats,
QueryInfo,
RtkQueryApiState,
RtkQueryTag,
SelectorsSource,
RtkQueryProvided,
QueryPreviewTabs,
RtkResourceInfo,
} from './types';
import { Comparator, queryComparators } from './utils/comparators';
import { FilterList, queryListFilters } from './utils/filters';
import { emptyRecord } from './utils/object';
import { escapeRegExpSpecialCharacter } from './utils/regexp';
import {
getApiStatesOf,
extractAllApiQueries,
flipComparator,
getQueryTagsOf,
generateApiStatsOfCurrentQuery,
getActionsOfCurrentQuery,
extractAllApiMutations,
} from './utils/rtk-query';
type InspectorSelector<S, Output> = Selector<SelectorsSource<S>, Output>;
export function computeSelectorSource<S, A extends Action<unknown>>(
props: RtkQueryInspectorProps<S, A>,
previous: SelectorsSource<S> | null = null
): SelectorsSource<S> {
const { computedStates, currentStateIndex, monitorState, actionsById } =
props;
const userState =
computedStates.length > 0 ? computedStates[currentStateIndex].state : null;
if (
!previous ||
previous.userState !== userState ||
previous.monitorState !== monitorState ||
previous.actionsById !== actionsById
) {
return {
userState,
monitorState,
currentStateIndex,
actionsById,
};
}
return previous;
}
export interface InspectorSelectors<S> {
readonly selectQueryComparator: InspectorSelector<S, Comparator<QueryInfo>>;
readonly selectApiStates: InspectorSelector<
S,
ReturnType<typeof getApiStatesOf>
>;
readonly selectAllQueries: InspectorSelector<
S,
ReturnType<typeof extractAllApiQueries>
>;
readonly selectAllVisbileQueries: InspectorSelector<S, RtkResourceInfo[]>;
readonly selectCurrentQueryInfo: InspectorSelector<S, RtkResourceInfo | null>;
readonly selectSearchQueryRegex: InspectorSelector<S, RegExp | null>;
readonly selectCurrentQueryTags: InspectorSelector<S, RtkQueryTag[]>;
readonly selectApiStatsOfCurrentQuery: InspectorSelector<S, ApiStats | null>;
readonly selectApiOfCurrentQuery: InspectorSelector<
S,
RtkQueryApiState | null
>;
readonly selectTabCounters: InspectorSelector<
S,
Record<QueryPreviewTabs, number>
>;
readonly selectSubscriptionsOfCurrentQuery: InspectorSelector<
S,
RtkQueryApiState['subscriptions'][string]
>;
readonly selectActionsOfCurrentQuery: InspectorSelector<
S,
ReturnType<typeof getActionsOfCurrentQuery>
>;
}
export function createInspectorSelectors<S>(): InspectorSelectors<S> {
const selectQueryComparator = ({
monitorState,
}: SelectorsSource<S>): Comparator<RtkResourceInfo> => {
return queryComparators[monitorState.queryForm.values.queryComparator];
};
const selectQueryListFilter = ({
monitorState,
}: SelectorsSource<S>): FilterList<RtkResourceInfo> => {
return queryListFilters[monitorState.queryForm.values.queryFilter];
};
const selectActionsById = ({ actionsById }: SelectorsSource<S>) =>
actionsById;
const selectApiStates = createSelector(
({ userState }: SelectorsSource<S>) => userState,
getApiStatesOf
);
const selectAllQueries = createSelector(
selectApiStates,
extractAllApiQueries
);
const selectAllMutations = createSelector(
selectApiStates,
extractAllApiMutations
);
const selectSearchQueryRegex = createSelector(
({ monitorState }: SelectorsSource<S>) =>
monitorState.queryForm.values.searchValue,
({ monitorState }: SelectorsSource<S>) =>
monitorState.queryForm.values.isRegexSearch,
(searchValue, isRegexSearch) => {
if (searchValue) {
try {
const regexPattern = isRegexSearch
? searchValue
: escapeRegExpSpecialCharacter(searchValue);
return new RegExp(regexPattern, 'i');
} catch (err) {
// We notify that the search regex provided is not valid
}
}
return null;
}
);
const selectComparatorOrder = ({ monitorState }: SelectorsSource<S>) =>
monitorState.queryForm.values.isAscendingQueryComparatorOrder;
const selectAllVisbileQueries = createSelector(
[
selectQueryComparator,
selectQueryListFilter,
selectAllQueries,
selectAllMutations,
selectComparatorOrder,
selectSearchQueryRegex,
],
(
comparator,
queryListFilter,
queryList,
mutationsList,
isAscending,
searchRegex
) => {
const filteredList = queryListFilter(
searchRegex,
(queryList as RtkResourceInfo[]).concat(mutationsList)
);
const computedComparator = isAscending
? comparator
: flipComparator(comparator);
return filteredList.slice().sort(computedComparator);
}
);
const selectCurrentQueryInfo = createSelector(
selectAllQueries,
selectAllMutations,
({ monitorState }: SelectorsSource<S>) => monitorState.selectedQueryKey,
(allQueries, allMutations, selectedQueryKey) => {
if (!selectedQueryKey) {
return null;
}
let currentQueryInfo: null | RtkResourceInfo =
allQueries.find(
(query) =>
query.queryKey === selectedQueryKey.queryKey &&
selectedQueryKey.reducerPath === query.reducerPath
) || null;
if (!currentQueryInfo) {
currentQueryInfo =
allMutations.find(
(mutation) =>
mutation.queryKey === selectedQueryKey.queryKey &&
selectedQueryKey.reducerPath === mutation.reducerPath
) || null;
}
return currentQueryInfo;
}
);
const selectApiOfCurrentQuery: InspectorSelector<S, null | RtkQueryApiState> =
(selectorsSource: SelectorsSource<S>) => {
const apiStates = selectApiStates(selectorsSource);
const currentQueryInfo = selectCurrentQueryInfo(selectorsSource);
if (!apiStates || !currentQueryInfo) {
return null;
}
return apiStates[currentQueryInfo.reducerPath] ?? null;
};
const selectProvidedOfCurrentQuery: InspectorSelector<
S,
null | RtkQueryProvided
> = (selectorsSource: SelectorsSource<S>) => {
return selectApiOfCurrentQuery(selectorsSource)?.provided ?? null;
};
const selectSubscriptionsOfCurrentQuery = createSelector(
[selectApiOfCurrentQuery, selectCurrentQueryInfo],
(apiState, queryInfo) => {
if (!queryInfo || !apiState) {
return emptyRecord;
}
return apiState.subscriptions[queryInfo.queryKey];
}
);
const selectCurrentQueryTags = createSelector(
[selectCurrentQueryInfo, selectProvidedOfCurrentQuery],
getQueryTagsOf
);
const selectApiStatsOfCurrentQuery = createSelector(
selectApiOfCurrentQuery,
(selectorsSource) => selectorsSource.actionsById,
(selectorsSource) => selectorsSource.currentStateIndex,
generateApiStatsOfCurrentQuery
);
const selectActionsOfCurrentQuery = createSelector(
selectCurrentQueryInfo,
selectActionsById,
getActionsOfCurrentQuery
);
const selectTabCounters = createSelector(
[
selectSubscriptionsOfCurrentQuery,
selectActionsOfCurrentQuery,
selectCurrentQueryTags,
],
(subscriptions, actions, tags) => {
return {
[QueryPreviewTabs.queryTags]: tags.length,
[QueryPreviewTabs.querySubscriptions]: Object.keys(subscriptions ?? {})
.length,
[QueryPreviewTabs.apiConfig]: 0,
[QueryPreviewTabs.queryinfo]: 0,
[QueryPreviewTabs.actions]: actions.length,
};
}
);
return {
selectQueryComparator,
selectApiStates,
selectAllQueries,
selectAllVisbileQueries,
selectSearchQueryRegex,
selectCurrentQueryInfo,
selectCurrentQueryTags,
selectApiStatsOfCurrentQuery,
selectSubscriptionsOfCurrentQuery,
selectApiOfCurrentQuery,
selectTabCounters,
selectActionsOfCurrentQuery,
};
}