deep-modules

📁 codybrom/clairvoyance 📅 1 day ago
4
总安装量
3
周安装量
#53536
全站排名
安装命令
npx skills add https://github.com/codybrom/clairvoyance --skill deep-modules

Agent 安装分布

opencode 3
claude-code 3
github-copilot 3
codex 3
kimi-cli 3
gemini-cli 3

Skill 文档

Deep Modules Review Lens

When invoked with $ARGUMENTS, focus the analysis on the specified file or module. Read the target code first, then apply the checks below.

Evaluate whether modules provide powerful functionality through simple interfaces.

When to Apply

  • Reviewing a new class, module, or API design
  • When a module feels like it “doesn’t do enough” or has too many parameters
  • When you see many small classes collaborating to do one thing
  • When a method signature closely mirrors what it calls
  • During refactoring to decide what to merge or deepen

Core Principles

The Depth Principle

“The benefit provided by a module is its functionality. The cost of a module (in terms of system complexity) is its interface.” — John Ousterhout, A Philosophy of Software Design

Deep modules maximize benefit relative to cost.

“It is more important for a module to have a simple interface than a simple implementation.” — John Ousterhout, A Philosophy of Software Design

Implementation complexity is paid once. Interface complexity is paid by every caller, forever. The deepest possible module has no interface at all. Garbage collection does complex work and callers never interact with it.

Remember: the interface has two layers. The formal interface (signatures, types, exceptions) and the informal interface (ordering constraints, side effects, performance characteristics, thread safety). The informal interface is usually larger and more dangerous. If undocumented, it produces unknown unknowns.

The Depth Test

For each module under review:

  1. How many things must a caller know to use this correctly?
  2. How much work does the module do behind that interface?
  3. Could a caller skip this module and do the work directly with similar effort?

If #3 is “yes,” the module is shallow.

A concrete signal: if the documentation for a method would be longer than its implementation, it is shallow.

Depth Applies at Every Scale

Methods Should Be Deep Too

A method with hundreds of lines is fine if it has a simple signature and does one complete thing. Splitting into shallow pieces replaces one interface with many: net cost increase. Depth first, length second.

“Don’t sacrifice depth for length.” — John Ousterhout, A Philosophy of Software Design

This directly contradicts the Clean Code position that functions should be broken up by length alone. Shorter is generally easier to read, but once a function is down to a few dozen lines, further reduction is unlikely to help. The real question is whether splitting reduces the complexity of the system, not the function in isolation. If the pieces lose their independence and must be read together, the split made things worse.

When Long Methods Are Fine

A method with several sequential blocks is fine as one method if the blocks are relatively independent.

“If the blocks have complex interactions, it’s even more important to keep them together so readers can see all of the code at once.” — John Ousterhout, A Philosophy of Software Design

Classitis

The dominant modern error is over-decomposing: splitting things into too many small pieces rather than too few large ones. You can often make something deeper by combining closely related things. Once unified, they may simplify each other in ways that were invisible when apart.

Signs:

  • Many classes each doing one small thing, requiring callers to compose them
  • Class names that are verbs or single operations (Reader, Writer, Parser, Validator as thin wrappers)
  • Deep dependency chains where each layer adds minimal abstraction
  • Needing 3+ objects to do one logical operation
  • Decorator chains where each layer adds one behavior and nineteen pass-throughs

The Defaults Principle

If nearly every user of a class needs a behavior, that behavior belongs inside the class by default, not in a separate wrapper. Merge related shallow classes into fewer, deeper ones. A class with 500 lines and 3 public methods is better than 5 classes with 100 lines and 15 total public methods. The question is never “is this class too large?” but “does this split reduce the total cognitive load on callers?”

Some modules are unavoidably shallow, like how a linked list class hides very few details behind its interface. These are still useful, but they don’t provide much leverage against complexity. Don’t mistake “shallow and acceptable” for “deep enough.”

Decorator Alternatives

Before creating a decorator class, consider:

  1. Add the functionality directly to the underlying class (if general-purpose or used by most callers)
  2. Merge the functionality with the specific use case instead of wrapping
  3. Merge with an existing decorator to create one deeper decorator instead of multiple shallow ones
  4. Implement as a standalone class independent of the base class

Pass-Through Method Audit

A pass-through method adds interface cost with zero benefit. The telltale sign is a wrapper class where most public methods just forward to an inner dependency (a membrane, not a layer). If removing the wrapper changes nothing about the caller’s experience, the class has no reason to exist.

Detection

  • Signature closely resembles the method it calls
  • Body is mostly a single delegation call
  • Removing the method wouldn’t lose any logic

Fixes (in Order of Preference)

  1. Expose the lower-level method directly
  2. Redistribute functionality so each layer has a distinct job
  3. Merge the classes

When Legitimate

Dispatchers that route based on arguments, and multiple implementations of a shared interface and duplicate signatures with meaningfully different behavior.

Pass-Through Variables

Data threaded through a chain of signatures just to reach one deep method. Each intermediate method must be aware of the variable without using it, and adding a new pass-through variable forces changes to every signature in the chain.

Fixes (in Order of Preference)

  1. Shared object: if an object is already accessible to both the top and bottom methods, store the data there
  2. Context object: a single object holding cross-cutting state (configuration, shared subsystems, performance counters), with references stored in instance variables via constructors so it doesn’t itself become a pass-through argument
  3. Global variable: avoids pass-through but prevents multiple instances of the system in one process. Generally avoid

Context objects are the most common solution but not ideal. Without discipline they become grab-bags of data that create non-obvious dependencies. Variables in a context should be immutable when possible.

Interface vs Implementation

If callers see the same structure they’d see without the class, the class isn’t hiding anything.

Shallow — config.getString("timeout"): callers must know the key name, parse the string to a number, and handle missing values themselves. The interface mirrors the internal key-value store.

Deep — config.getTimeout(): callers get a typed value with defaults already applied. The class absorbs parsing, validation, and key naming so callers don’t have to.

Interface Simplification Tactics

  • Defaults: Every parameter with a sensible default is one less thing callers must specify. The common case should require no special effort from callers
  • Define errors out of existence: Every eliminated exception is one less interface element
  • General-purpose design: Distill to the essential operation. Special cases go in a layer above
  • Pull complexity downwards: Handle edge cases internally rather than exposing them

Review Process

  1. List modules: Identify the classes, functions, or APIs under review
  2. Measure depth: For each: count interface elements vs. implementation scope
  3. Flag shallow modules: Any module where interface ~= implementation in complexity
  4. Check for classitis: Are there clusters of thin classes that should be merged?
  5. Audit pass-throughs: Are any methods just forwarding calls?
  6. Propose deepening: For each issue, recommend: merge, absorb, or eliminate

Relationship to Other Lenses

This skill asks “is the module deep enough?”: does the interface justify what’s behind it? abstraction-quality asks the prior question: “is the abstraction genuine?”: does each layer provide a different way of thinking? A module can be deep but sit in a layer that duplicates the abstraction of its neighbor. When adjacent layers look suspiciously similar, hand off to abstraction-quality.

Red flag signals for module depth are cataloged in red-flags (Shallow Module, Pass-Through Method).