Domain 2Task 2.5

Built-in Tools

Learn this interactively

What You Need to Know

Claude Code provides six built-in tools for working with codebases: Read, Write, Edit, Bash, Grep, and Glob. Each has a specific purpose, and using the wrong tool for a task wastes time, context tokens, or both. The exam deliberately presents scenarios where confusing these tools leads to incorrect answers.

Grep vs Glob: The Core Distinction

This is the single most important distinction in this task statement. Get this wrong and you will lose marks.

Grep searches file CONTENTS for patterns. Use Grep when you need to find text inside files. Function callers. Error messages. Import statements. Variable assignments. Any time you are searching for what files contain, Grep is the tool.

// Find all files that call processLegacyOrder() Grep: "processLegacyOrder" // Find all error messages containing "timeout" Grep: "timeout" // Find all files that import a specific module Grep: "import.*from 'utils/auth'"

Glob matches file PATHS by naming patterns. Use Glob when you need to find files by name, extension, or directory structure. Test files. Configuration files. All TypeScript files in a specific directory. Any time you are searching for files based on their path, Glob is the tool.

// Find all test files Glob: "**/*.test.tsx" // Find all configuration files Glob: "**/config.*" // Find all MDX files in the domains directory Glob: "content/domains/**/*.mdx"

The distinction in one sentence: Grep finds what is INSIDE files. Glob finds files by their NAMES.

The exam presents scenarios where a developer uses the wrong tool. If someone uses Glob to find function callers, it will fail — Glob matches paths, not contents. If someone uses Grep to find test files by naming pattern, it will work technically (by searching for "test" in filenames via content), but it is the wrong tool and the exam expects you to identify the correct one.

Read, Write, and Edit

These three tools handle file operations, each optimised for a different use case.

Edit performs targeted modifications using unique text matching. You specify the exact text to find and its replacement. It is fast and precise because it touches only the specific text you identify.

Edit: old_string: "function processOrder(id: string)" new_string: "function processOrder(id: string, validate: boolean = true)"

When Edit fails: Edit requires unique text matching. If the text you specify appears in multiple places in the file, Edit cannot determine which occurrence to change and will fail. This is not a bug — it is a safety mechanism preventing unintended modifications.

When Edit can't find a unique anchor. The fix per the Edit tool docs is to widen old_string with more surrounding context until it pins down one location, or set replace_all: true if you actually want every occurrence updated. Both options keep you on Edit and cost almost no extra context. Read + Write (load the whole file, write the whole file back) is the last resort. It works. But you've now spent a file's worth of tokens on what was usually a one-line change.

The ordering:

  1. Try Edit with the shortest anchor that's plausibly unique.
  2. On a non-unique match, widen old_string until it matches one location, or use replace_all: true if you want every occurrence changed.
  3. Only fall back to Read + Write when neither of those can disambiguate the target.

Don't default to Read + Write for every modification. The exam penalises that because it burns context tokens. It also penalises jumping straight from a non-unique Edit failure to Read + Write. Widening the anchor or using replace_all is the documented response. Read + Write is the fallback, not the next step.

Incremental Codebase Understanding

How you explore a codebase matters as much as which tools you use. There is a right way and a wrong way.

Wrong: Read all files upfront. Loading every file into context before you understand what you need is a context-budget killer. If a codebase has 200 files and you read them all, you have consumed your entire context window on files that are mostly irrelevant to your task. This is the single biggest anti-pattern in codebase exploration.

Right: Incremental discovery. Start narrow. Expand only as needed.

  1. Grep to find entry points. Search for the function name, class name, or error message that anchors your investigation. This tells you which files are relevant.

  2. Read to follow imports and trace flows. Once you know which files matter, Read them to understand the code structure. Follow import statements to discover related files.

  3. Grep again to trace usage. If you find a wrapper function or re-export, Grep for that name across the codebase to find all consumers.

  4. Read only what you need. Each file you read should be justified by what you discovered in the previous step.

This approach uses minimal context for maximum understanding. You discover the codebase's structure progressively, spending tokens only on files that are relevant to your task.

Tracing Function Usage Across Wrapper Modules

A common codebase pattern: a function is defined in one module, re-exported through a wrapper, and consumed through the wrapper's name. Simple Grep for the original function name will miss consumers who import through the wrapper.

The correct approach:

  1. Grep for the function definition to find where it is defined
  2. Read the defining file to identify exported names
  3. Grep for each exported name across the codebase to find all consumers
  4. If the function is re-exported through a barrel file (e.g. index.ts), Grep for the barrel file's module name to find consumers who import from it

This multi-step trace is more thorough than a single Grep and catches indirect consumers that a simple search would miss.

The Deprecation Scenario

This scenario appears frequently in exam preparation: find all files that call a deprecated function AND find the test files that exercise it. The correct tool sequence:

  1. Grep for the function name — finds every file whose contents reference the function, including any tests that import it directly (content search)
  2. Glob for sibling test files — finds the test file that pairs with each caller by naming convention, e.g. OrderProcessor.tsOrderProcessor.test.tsx, even when the test exercises the function indirectly through the source module (path matching)
  3. Grep again for wrapper names — when a caller exposes the function through a wrapper (e.g. applyLegacyOrder calls processLegacyOrder internally), Grep for the wrapper name to find tests that cover the function transitively through it

For example, if Grep reveals that OrderProcessor.ts and RefundHandler.ts call the deprecated function, Glob for **/OrderProcessor.test.* and **/RefundHandler.test.* to pull in their sibling test files even if those tests never mention processLegacyOrder by name. If either source file wraps the function under a new name, Grep for that wrapper name to catch any additional tests.

This is Grep, then Glob, then Grep again — content search for direct references, path matching for adjacent tests, content search for indirect coverage. Not Glob first.

Key Concept

Grep searches file contents. Glob matches file paths. Edit is the default for modifications. On a non-unique match, widen the anchor or use replace_all: true. Read + Write is the last-resort fallback. Build codebase understanding incrementally. Never read all files upfront.

Exam Traps

Exam Trap

Using Glob to find function callers (it searches paths, not contents)

Glob matches file paths by naming pattern. It cannot search inside files for function calls. Use Grep to search file contents for function names, import statements, or error messages.

Exam Trap

Using Grep to find files by extension or naming pattern

While Grep could technically find filenames mentioned in content, Glob is the purpose-built tool for matching file paths. Use Glob for **/*.test.tsx, **/config.*, and similar path-based searches.

Exam Trap

Reading all source files upfront before understanding what is relevant

Loading every file into context is a context-budget killer. The correct approach is incremental: Grep to find entry points, then Read to trace flows from those specific entry points.

Exam Trap

Defaulting to Read + Write for every file modification instead of trying Edit first

Edit is faster and uses less context because it only touches the specific text. Read + Write loads the entire file. Try Edit first; widen the anchor or use replace_all when Edit reports a non-unique match. Read + Write is the last-resort fallback, not the standard response.

Exam Trap

Jumping straight to Read + Write the moment Edit reports a non-unique match

The documented Edit recovery is to widen old_string with surrounding context until it pins down one occurrence, or to set replace_all: true for a global change. Both stay on Edit and cost almost nothing. Escalate to Read + Write only when neither option can disambiguate the target.

Practice Scenario

A developer needs to find all files that call a deprecated function processLegacyOrder() and also find all test files for those callers. Which tool sequence is correct?

Build Exercise

Build Exercise

Trace and Refactor a Deprecated Function Using Built-in Tools

Difficulty
30 minutes

What you'll learn

  • Apply Grep for content search and Glob for path matching in the correct sequence
  • Use incremental codebase discovery instead of reading all files upfront
  • Select Edit as the primary modification tool and widen the anchor (or use replace_all) when Edit reports a non-unique match
  • Trace function usage across wrapper modules and barrel files
  • Follow the Grep-then-Glob pattern for finding callers and their test files
  1. Use Grep to search for all callers of a target function (e.g. processLegacyOrder) across the codebase

    Why: Grep searches file contents — it is the correct tool for finding function callers. Using Glob here would fail because Glob matches file paths, not contents. The exam tests this distinction directly and penalises candidates who confuse the two.

    You should see: A list of file paths containing calls to processLegacyOrder, with line numbers and matching lines showing the exact call sites. For example: src/OrderProcessor.ts:42: await processLegacyOrder(orderId).

  2. Use Glob to find test files matching the caller filenames (e.g. **/*.test.tsx)

    Why: Glob matches file paths by naming pattern — it is the correct tool for finding test files by extension or naming convention. This completes the Grep-then-Glob pattern: content search to find callers, then path matching to find their tests.

    You should see: A list of test file paths matching the pattern, such as src/OrderProcessor.test.tsx and src/RefundHandler.test.tsx. These correspond to the caller files found by Grep in the previous step.

  3. Use Read to examine each caller file and understand the usage pattern and context

    Why: Reading files incrementally — only after Grep identifies which files matter — is the correct approach. Reading all source files upfront is a context-budget killer that the exam explicitly penalises. Each Read should be justified by what you discovered in the previous step.

    You should see: The full contents of each caller file, showing how processLegacyOrder is called, what parameters are passed, how the return value is used, and whether the function is imported directly or through a wrapper module.

  4. Use Edit to replace the deprecated function call with the new API in each caller file

    Why: Edit is the preferred modification tool because it targets specific text and uses less context than Read + Write. The exam penalises defaulting to Read + Write for every modification. Always try Edit first — it is faster and more precise.

    You should see: Each caller file updated with the new API call replacing the deprecated one. For example, processLegacyOrder(orderId) replaced with processOrder(orderId, { validate: true }). The Edit tool confirms the replacement was made successfully.

  5. When Edit fails with a non-unique match, widen old_string with more surrounding lines until it pins down one location (or set replace_all: true if you actually want every occurrence updated). Only fall back to Read + Write if neither option can disambiguate the target

    Why: Edit fails when the target text appears multiple times in the file — this is a safety mechanism, not a bug. Per the Edit tool documentation, the documented recovery is to expand the anchor with more surrounding context until it matches one place, or to use replace_all for global replacements. Both keep you on Edit and cost almost nothing in context. Read + Write loads the entire file for what is usually a single-line change — keep it as a last resort.

    You should see: On the first try, Edit fails with an error like: old_string matches 3 locations. On the retry with a wider old_string that includes the surrounding function name or unique adjacent line, Edit succeeds and changes exactly one occurrence. If replace_all: true was the right call, every occurrence is updated atomically.

Sources