mirror of
https://github.com/reduxjs/redux-devtools.git
synced 2024-11-25 19:13:56 +03:00
7d92a5e186
* 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
284 lines
7.9 KiB
TypeScript
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,
|
|
};
|
|
}
|