Most developers believe that setting "strict": true in their _tsconfig.json_ is the ultimate safety net for their codebase. It is a fantastic starting point, but relying on it alone leaves the door wide open for catastrophic type omissions and hidden runtime bugs.
You can have a file completely riddled with type errors without a single warning or red underline from your compiler. To achieve true type safety and completely eliminate silent failures, you need to look beyond strict mode.
These are the 5 highly effective _tsconfig.json_ configurations will harden your type-checking system, catch developer typos before they hit production, and dramatically streamline your development workflow.
Table of Contents
- 1. The Workflow Game-Changer: Path Aliases
- 2. The Code-Cleaners: Eliminating Dead Weight
- 3. The Typo-Catchers: Stopping Silent Failures
- 4. Optional but Highly Recommended Enhancements
- 5. Personal Preference & Architecture Tuning
- 6. Seamless JavaScript Migration Settings
- Summary Checklist
1. The Workflow Game-Changer: Path Aliases
This first configuration option does not alter type checking, but it fundamentally transforms how you interact with your project's codebase.
The Problem: Relative Import Hell
As a project grows, directory structures naturally deepen. When deep nesting occurs, importing a module from the root or a shared directory turns into a fragile, unreadable mess of relative paths:
// Naive / Problematic Approach
import { RootConfig } from '../../../../../config/root';
import { Button } from '../../../../components/ui/Button';
If you move or copy this file to another directory, every single relative import breaks, forcing a tedious manual reconfiguration.
The Solution: Absolute Imports via paths
By configuring paths in your tsconfig.json, you can define custom absolute alias structures.
{
"compilerOptions": {
"baseUrl": ".",
"paths": {
"@/*": ["src/*"],
"@components/*": ["src/components/*"]
}
}
}
Note: The baseUrl property must be specified whenever using paths so TypeScript has a benchmark directory to resolve your aliases against.
With this alias map established, your import statements become completely location-agnostic and clean:
// Optimized / Best-Practice Approach
import { RootConfig } from '@/config/root';
import { Button } from '@components/ui/Button';
2. The Code-Cleaners: Eliminating Dead Weight
Unused variables and dead parameters clutter codebases, mask architectural intent, and frequently point to uncompleted refactoring or underlying typos.
{
"compilerOptions": {
"noUnusedLocals": true,
"noUnusedParameters": true
}
}
noUnusedLocals
When set to true, the compiler flags any local variable declared within your code that is never read or executed.
noUnusedParameters
Similarly, this flags any function or method parameter that goes unused within its respective scope.
// Triggers compiler errors for unused variables and parameters
function calculateTotal(price: number, taxRate: number, discountCode: string): number {
// Error: 'discountCode' is declared but its value is never read.
const localTax = price * taxRate;
const legacyMarkup = 10; // Error: 'legacyMarkup' is declared but never read.
return price + localTax;
}
Enabling these options keeps your production code clean, easily human-readable, and free from vestigial variables.
3. The Typo-Catchers: Stopping Silent Failures
These options protect you from esoteric language quirks, silent file resolution issues, and logic holes.
{
"compilerOptions": {
"allowUnusedLabels": false,
"noUncheckedSideEffectImports": true,
"noFallthroughCasesInSwitch": true,
"allowUnreachableCode": false
}
}
allowUnusedLabels
JavaScript has an esoteric, rarely used feature called labels that functions like a structural goto statement. It is a historical pattern that is almost always written by mistake instead of an object literal.
// Naive / Problematic Approach (Valid JS, but highly error-prone)
function createUser() {
const user = {
id: 'usr_100'
};
// Typo: Intended to assign 'name' inside the user object.
// Instead, this creates a dangling JavaScript label.
name: 'John Doe';
}
Setting "allowUnusedLabels": false catches this syntax immediately, flagging the mistake as a compile error.
noUncheckedSideEffectImports
Sometimes you need to import a module solely for its side effects (e.g., polyfills, CSS frameworks, global monitoring setups) rather than binding specific components:
import './analytics'; // Side-effect import
By default, if you misspell this filename (e.g., import './analytic'), standard bundlers and TypeScript compilers may fail to throw an explicit error at build time. Enabling noUncheckedSideEffectImports forces TypeScript to actively verify the physical disk existence of side-effect targets, preventing silent deployment failures.
noFallthroughCasesInSwitch
Without explicit termination, switch statement cases cascade sequentially into one another. Unless you intentionally omit break statements to share logic, this behavior yields severe runtime bugs.
// Switch fallthrough error example
type Status = 'success' | 'error' | 'pending';
function handleStatus(status: Status) {
switch (status) {
case 'success': // Error: Fallthrough case in switch.
logSuccess();
// Missing break statement!
case 'error':
logError();
break;
}
}
If you intentionally want multiple cases to run the exact same logic block, you can safely structure them consecutively without a body:
// Correct multi-case handling
switch (status) {
case 'success':
case 'pending':
processData(); // Runs for both success and pending
break;
case 'error':
logError();
break;
}
allowUnreachableCode
Any execution block residing past an explicit return, throw, or break statement can never be reached. Turning "allowUnreachableCode": false helps you proactively sweep away zombie code.
4. Optional but Highly Recommended Enhancements
These advanced settings require small structural changes to how you write your code, but they introduce an unprecedented layer of runtime safety.
{
"compilerOptions": {
"noUncheckedIndexAccess": true,
"noPropertyAccessFromIndexSignature": true
}
}
noUncheckedIndexAccess
This is one of the most powerful options available in the compiler. Consider an array lookup:
const numbers: number[] = [1, 2, 3];
const tenthElement = numbers[9]; // TypeScript infers this type as 'number' by default
console.log(tenthElement.toFixed()); // Runtime Crash: Cannot read properties of undefined
TypeScript naturally assumes that accessing a valid index of a typed array instantly yields that type. It cannot guess the runtime length of your collection.
Enabling noUncheckedIndexAccess forces every array or index-signature lookup to implicitly append | undefined to the returned type signature:
const numbers: number[] = [1, 2, 3];
const tenthElement = numbers[9]; // Inferred Type: number | undefined
// Error: Object is possibly 'undefined'.
console.log(tenthElement.toFixed());
// Optimized / Best-Practice Approach
console.log(tenthElement?.toFixed());
noPropertyAccessFromIndexSignature
When building configuration containers, you often define open-ended index signatures to accept unpredictable developer keys:
type AppSettings = {
darkMode: boolean;
[customKey: string]: string | number | boolean;
};
By default, standard dot notation (settings.darkMod) allows arbitrary properties because the object explicitly accepts any key string. This makes it incredibly easy to accidentally introduce a subtle typo on your core settings.
Enabling noPropertyAccessFromIndexSignature explicitly blocks dot notation access for anything not explicitly hard-coded in the type schema:
const settings: AppSettings = { darkMode: true, apiEndpoint: '/v1' };
// Error: Property 'darkMod' does not exist on type 'AppSettings'.
const isDark = settings.darkMod;
// Accessing dynamic, open-ended entries requires explicit bracket notation
const endpoint = settings['apiEndpoint'];
5. Personal Preference & Architecture Tuning
These configurations adapt to specific architectural preferences, such as Object-Oriented Programming (OOP) paradigms or deep debugging environments.
{
"compilerOptions": {
"noImplicitOverride": true,
"noErrorTruncation": false,
"exactOptionalPropertyTypes": true
}
}
noImplicitOverride
If you design your codebase around class inheritance, overriding parent methods without explicit markings can make code maintenance incredibly difficult.
class ViewComponent {
close() { /* parent logic */ }
}
class ModalWindow extends ViewComponent {
// Error: This member must have an 'override' modifier
// because it overrides a member in the base class 'ViewComponent'.
close() { /* child logic */ }
}
Adding the explicit keyword ensures that your architectural intent is clear and protected:
class ModalWindow extends ViewComponent {
override close() { /* child logic */ }
}
noErrorTruncation
When dealing with deeply nested generics or complex utility frameworks (like Zod, Drizzle, or TanStack Query), TypeScript's error tooltips can become incredibly long. By default, the compiler cuts these messages off with an ellipsis (...).
Setting "noErrorTruncation": true instructs the compiler to print out the full, unabridged type structure inside error logs.
Tip: Keep this disabled by default to avoid overwhelming error blocks, and toggle it on temporarily only when diagnosing complex type puzzles.
exactOptionalPropertyTypes
In vanilla JavaScript, an optional property is structurally ambiguous. There is a distinct difference between a property being omitted entirely and a property being explicitly set to undefined.
type UserProfile = {
bio?: string;
};
// Without exactOptionalPropertyTypes, both are allowed:
const userA: UserProfile = {}; // bio is missing
const userB: UserProfile = { bio: undefined }; // bio exists, holding undefined value
Why does this distinction matter? Operators like 'bio' in user yield completely different boolean values depending on whether the key exists. Turning on exactOptionalPropertyTypes enforces strict precision: an optional property can be omitted, but it cannot be explicitly assigned undefined.
6. Seamless JavaScript Migration Settings
If you are migrating an older JavaScript application to TypeScript, changing your whole project overnight is rarely an option. These tools enable a gradual, file-by-file adoption process.
{
"compilerOptions": {
"allowJs": true,
"checkJs": false
}
}
allowJs
Instructs the TypeScript engine to parse and import plain .js files alongside your native .ts structures instead of throwing immediate import resolution errors.
checkJs
When turned on globally, TypeScript applies its full type-checking logic to your legacy JavaScript modules using inferred type algorithms and JSDoc declarations.
Recommended Migration Strategy
Rather than enabling checkJs globally and instantly generating thousands of errors across an enterprise app, leave "checkJs": false in your master config. Instead, append the // @ts-check directive to the very top line of individual JavaScript files as you refactor them:
// @ts-check
import { parseAmount } from './utils';
// TypeScript checks this file and catches the incorrect type assignment!
// Error: Argument of type 'number' is not assignable to parameter of type 'string'.
parseAmount(1250);
Summary Checklist
Here is a look at a comprehensive, high-security tsconfig.json template ready to drop into your root directory:
{
"compilerOptions": {
"target": "ES2022",
"module": "NodeNext",
"moduleResolution": "NodeNext",
"strict": true, /* The baseline foundation */
/* Absolute Path Maps */
"baseUrl": ".",
"paths": {
"@/*": ["src/*"]
},
/* Code Cleaners */
"noUnusedLocals": true,
"noUnusedParameters": true,
/* Typo Prevention & Logic Safeguards */
"allowUnusedLabels": false,
"noUncheckedSideEffectImports": true,
"noFallthroughCasesInSwitch": true,
"allowUnreachableCode": false,
/* Advanced Type Safety Rules */
"noUncheckedIndexAccess": true,
"noPropertyAccessFromIndexSignature": true,
"exactOptionalPropertyTypes": true,
/* Code Consistency */
"noImplicitOverride": true
}
}
Final Thoughts 💡
What would be your approach ?
Are you already running strict rules like noUncheckedIndexAccess in your current applications, or are you about to add them to your configurations for the first time? If you have a favorite compiler flag that saved your code from a production incident, drop a comment below!
If you find errors in this article please drop a comment too. Thanks for your time !


















