diff --git a/src/hooks/useSelector.ts b/src/hooks/useSelector.ts index 0d16367ec..e9a34838f 100644 --- a/src/hooks/useSelector.ts +++ b/src/hooks/useSelector.ts @@ -105,6 +105,12 @@ export function createSelectorHook( ) { const toCompare = selector(state) if (!equalityFn(selected, toCompare)) { + let stack: string | undefined = undefined + try { + throw new Error() + } catch (e) { + ;({ stack } = e as Error) + } console.warn( 'Selector ' + (selector.name || 'unknown') + @@ -114,6 +120,7 @@ export function createSelectorHook( state, selected, selected2: toCompare, + stack, } ) } @@ -126,11 +133,18 @@ export function createSelectorHook( ) { // @ts-ignore if (selected === state) { + let stack: string | undefined = undefined + try { + throw new Error() + } catch (e) { + ;({ stack } = e as Error) + } console.warn( 'Selector ' + (selector.name || 'unknown') + ' returned the root state when called. This can lead to unnecessary rerenders.' + - '\nSelectors that return the entire state are almost certainly a mistake, as they will cause a rerender whenever *anything* in state changes.' + '\nSelectors that return the entire state are almost certainly a mistake, as they will cause a rerender whenever *anything* in state changes.', + { stack } ) } } diff --git a/test/hooks/useSelector.spec.tsx b/test/hooks/useSelector.spec.tsx index 81839b71e..d527c9c44 100644 --- a/test/hooks/useSelector.spec.tsx +++ b/test/hooks/useSelector.spec.tsx @@ -805,6 +805,7 @@ describe('React', () => { }), selected: expect.any(Number), selected2: expect.any(Number), + stack: expect.any(String), }) ) }) @@ -920,7 +921,10 @@ describe('React', () => { ) expect(consoleSpy).toHaveBeenCalledWith( - expect.stringContaining('returned the root state when called.') + expect.stringContaining('returned the root state when called.'), + expect.objectContaining({ + stack: expect.any(String), + }) ) }) })