-
Notifications
You must be signed in to change notification settings - Fork 13.1k
Description
π Search Terms
All searched terms
"jest"
"mock"
"vitest"
"filename"
"rename" (this is NOT a rename default export when file name changes request)
β Viability Checklist
- This wouldn't be a breaking change in existing TypeScript/JavaScript code
- This wouldn't change the runtime behavior of existing JavaScript code
- This could be implemented without emitting different JS based on the types of the expressions
- This isn't a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, new syntax sugar for JS, etc.)
- This isn't a request to add a new utility type: https://github.com/microsoft/TypeScript/wiki/No-New-Utility-Types
- This feature would agree with the rest of our Design Goals: https://github.com/Microsoft/TypeScript/wiki/TypeScript-Design-Goals
β Suggestion
When renaming or moving files in TypeScript projects, the language service should automatically update string literal paths in test framework mocking functions (e.g., jest.mock(), vitest.mock()) in addition to standard import statements.
Use Cases
When you rename/move a file, TypeScript updates imports but not mock paths:
// Before renaming helper.ts β utils/helper.ts
import { helper } from "./helper";
jest.mock("./helper"); // β Not updated - breaks tests!
// After rename
import { helper } from "./utils/helper"; // β
Updated automatically
jest.mock("./helper"); // β Still points to old location - test failsThis leads to:
- Broken tests after refactoring
- Manual find-and-replace operations
- Time wasted tracking down test failures
- More tedium when reorganizing code structure
Desired Behavior
After renaming/moving a file, both imports and mock paths should update:
// Before renaming helper.ts β utils/helper.ts
import { helper } from "./helper";
jest.mock("./helper");
// After rename
import { helper } from "./utils/helper"; // β
Updated
jest.mock("./utils/helper"); // β
Also updated!Examples
Jest Patterns
jest.mock("../services/api");
jest.unmock("../services/api");
jest.requireActual("../services/api");
jest.requireMock("../services/api");
jest.doMock("../services/api");
jest.dontMock("../services/api");Vitest Patterns
vitest.mock("../services/api");
vi.mock("../services/api");Dynamic Imports & Require
const module = await import("../services/api");
const module = require("../services/api");Expected Behavior
- When a file is renamed/moved via IDE operations (F2, drag-drop, etc.)
- TypeScript should scan all files for references to the old path
- Update all string literals that reference the file, including:
- Standard
import/exportstatements (already works) jest.mock()and related Jest functionsvitest.mock()and related Vitest functionsrequire()calls- Dynamic
import()expressions
- Standard
- Only update relative paths that actually point to the renamed file
- Skip non-local modules (npm packages)
Workarounds
Currently, developers must:
- Rename the file
- Manually find all test files that mock it
- Update each mock path by hand
- Hope they didn't miss any
Additional Context
- This affects large codebases more severely (more mocks = more manual work)
- Common in projects with comprehensive test coverage
- Similar features exist in other IDEs (e.g., WebStorm handles this)
- Would significantly improve refactoring experience in test-heavy projects
Proposed Solution
Add an opt-in preference (e.g., updateImportsInTestFrameworkCalls) that enables this behavior. The feature should:
- Be conservative (only update simple patterns)
- Be extensible (easy to add support for new test frameworks)
- Avoid false positives (only update actual file references)
- Work with both JavaScript and TypeScript files
Alignment with TypeScript Goals
This feature aligns with TypeScript's core design goals:
-
"Statically identify constructs that are likely to be errors"
- Outdated mock paths after file renames are errors that TypeScript can detect and prevent
- Currently these errors only surface at test runtime, this would catch them during development
-
"Provide a structuring mechanism for larger pieces of code"
- Enables confident refactoring and code organization without fear of breaking tests
- Makes it easier to maintain well-structured codebases by removing friction from file reorganization
-
"Be a cross-platform development tool"
- Enhances the IDE/editor experience across all platforms
- Improves the TypeScript language service that powers multiple editors
-
"Preserve runtime behavior of all JavaScript code"
- This is purely a development-time tooling enhancement
- No changes to emitted code or runtime behavior
- Only updates source code strings to maintain correct runtime behavior after refactoring
Affected Packages
- TypeScript Language Service
- VS Code TypeScript Extension (for preference exposure)
π Motivating Example
You're refactoring a large TypeScript project and decide to reorganize your file structure. You move src/api/userService.ts to src/services/users/userService.ts. TypeScript automatically updates all your imports:
// Before
import { getUser } from '../../api/userService';
// After
import { getUser } from '../../services/users/userService'; β
Great! But then your tests start failing. You discover that TypeScript didn't update your test mocks:
jest.mock('../../api/userService'); // β Still points to old location!You now have to manually search through dozens (or hundreds) of test files, find every jest.mock() call that references the moved file, and update each one by hand. This is tedious, error-prone, and makes refactoring feel risky.
With this feature, TypeScript would treat mock calls the same as imports:
jest.mock('../../services/users/userService'); // β
Updated automatically!Now you can confidently refactor your codebase knowing that both imports and mocks will stay in sync. Your tests keep passing, and you spend time improving code instead of hunting down broken references.
π» Use Cases
- What do you want to use this for?
Large refactors and more confidence in IDE tooling. - What shortcomings exist with current approaches?
This is not covered by the current version of TypeScript. - What workarounds are you using in the meantime?
When moving single files have to also search for all references to that files (or forget about that and have CI failing)
When moving folders I need to run the full suite to find where there might be file references that need a manual update.