Agent skill
holistic-linting-resolver
Linter-specific resolution workflows for ruff, mypy, pyright, and basedpyright. Provides systematic root-cause analysis procedures, suppression gates, and verification steps. Use when resolving linting errors as a sub-agent, implementing fixes systematically, or conducting type flow analysis.
Install this agent skill to your Project
npx add-skill https://github.com/Jamie-BitFlight/claude_skills/tree/main/plugins/holistic-linting/skills/holistic-linting-resolver
SKILL.md
Holistic Linting: Resolution Workflows
This skill provides systematic resolution procedures for Python linting tools. Sub-agents executing linting resolution tasks MUST follow the appropriate workflow based on the linter reporting issues.
When to Use This Skill
Use this skill when you are a sub-agent assigned to resolve linting issues in a specific file. This skill provides detailed workflows for:
- Ruff (style, import, and code quality checks)
- Mypy (type checking)
- Pyright/Basedpyright (strict type checking)
Do NOT use this skill if you are an orchestrator. Orchestrators should use the holistic-linting-orchestrator skill for delegation workflows.
Common Resolution Methodology
All linter-specific workflows share these common steps. Apply them in order before the linter-specific procedures.
1. Load python3-development Skill
Before implementing any fixes:
Skill(skill: "python3-development:python3-development")
Motivation: Ensures fixes follow Python 3.11+ standards, modern typing patterns, and project conventions.
2. Suppression Gate (MANDATORY)
Before implementing any fix, verify it is a code change — not suppression. Each category below is an immediate STOP:
Inline suppression comments:
- Adding
# noqa,# type: ignore,# pyright: ignore,# pylint: disable, or any suppression comment
Configuration-level suppression (equally forbidden):
- Adding a rule to
[tool.ruff.lint] ignore = [...]inpyproject.toml - Adding an entry to
[tool.ruff.lint.per-file-ignores] - Changing
[tool.pyright] reportX = "warning"or any other severity downgrade - Adding
disable_error_code = [...]to[tool.mypy] - Modifying any linter config file (
pyproject.toml,ruff.toml,mypy.ini,.flake8,setup.cfg) to reduce the scope, severity, or applicability of a rule
Reason: A pyproject.toml severity downgrade achieves the same silencing effect as # type: ignore but at project scope, affecting all future code. Both are standards degradation. Neither is a code fix.
Deletion-as-resolution (equally forbidden):
- Removing a function, class, method, or test to eliminate the linting error within it
- Deleting lines of code solely because they contain a linting error
If any proposed fix falls into these categories: STOP.
When no code restructuring works (minimum 2 approaches attempted), document the constraint and return UNRESOLVED:
- What you tried (each approach, minimum 2)
- Why each approach failed (exact linter error produced)
- The fundamental constraint (e.g., "ruff cannot evaluate
sys.platformbranches")
Return this as an UNRESOLVED item in your resolution report. The orchestrator will surface it to the user for a human decision on whether to suppress, reconfigure, or accept the limitation.
3. Check Architectural Context
Examine how this code fits into the broader system:
- What does this function/module do?
- How is it called by other code?
- Are there similar patterns elsewhere in the codebase?
Use Grep to find usage patterns:
uv run rg "function_name" --type py
4. Verify Resolution
After implementing fixes, rerun the appropriate linter on the primary file:
# For Ruff:
uv run ruff check /path/to/file.py
# For Mypy:
uv run mypy /path/to/file.py
# For Pyright/Basedpyright:
uv run pyright /path/to/file.py
# or
uv run basedpyright /path/to/file.py
Verify incidentally modified files: If you touched any file other than the primary target during the fix (e.g., added an import to a utility module, moved a function, updated a type alias), run the linter on those files too:
uv run ruff check /path/to/incidentally/modified/file.py
uv run mypy /path/to/incidentally/modified/file.py
All incidentally modified files must also produce zero errors before resolution is complete.
Record the before and after issue counts in the resolution report header:
**Issues before resolution:** N (from initial linter run)
**Issues after resolution:** 0 (from final linter run — must be 0 for all touched files)
**UNRESOLVED items:** N (must be explicitly listed even if 0)
Pre-existing issues detected: If the initial linter run reveals issues in files you did not touch — see Pre-Existing Issues Protocol. Every detected issue gets recorded. Silence is not permitted.
Ruff Resolution Workflow
When to use: Linting errors with ruff rule codes (E, F, W, B, S, I, UP, C90, N, etc.)
Resolution Process:
-
Research the Rule
Use ruff's built-in documentation system:
bashruff rule {RULE_CODE}Examples:
bashruff rule F401 # unused-import ruff rule E501 # line-too-long ruff rule B006 # mutable-default-argumentThis command provides:
- What the rule prevents (design principle)
- When code violates the rule
- Example of violating code
- Example of resolved code
- Configuration options
-
Read Rule Documentation Output
The ruff rule output contains critical information:
- Principle: Why this pattern is problematic
- Bad Pattern: What code triggers the rule
- Good Pattern: How to fix it correctly
Motivation: Understanding the principle prevents similar issues in other locations.
-
Read the Affected Code
Read the complete file containing the linting error:
claudeRead("/path/to/file.py")Focus on:
- The line with the error
- Surrounding context (5-10 lines before/after)
- Related function/class definitions
-
Apply Common Methodology
Follow steps 1-4 in the Common Resolution Methodology section above:
- Load python3-development skill
- Pass through Suppression Gate
- Check Architectural Context
- Verify Resolution
-
Implement Elegant Fix
Apply the fix following these principles:
- Address the root cause, not the symptom
- Follow modern Python patterns from python3-development skill
- Maintain or improve code readability
- Consider performance and maintainability
- Add comments only if the fix is non-obvious
Mypy Resolution Workflow
When to use: Type checking errors with mypy error codes (attr-defined, arg-type, return-value, etc.)
Resolution Process:
-
Research the Error Code
Mypy errors contain error codes in brackets like
[attr-defined]or[arg-type].Look up the error code in locally-cached documentation:
claudeRead("./references/mypy-docs/error_code_list.rst") Read("./references/mypy-docs/error_code_list2.rst")Search for the error code:
bashgrep -n "error-code-{CODE}" ./references/mypy-docs/*.rstMotivation: Mypy error codes map to specific type safety principles. Understanding the principle prevents misunderstanding type relationships.
-
Read Error Code Documentation
The mypy documentation explains:
- What type safety principle is violated
- When this is an error (type violations)
- When this is NOT an error (valid patterns)
- Example of error-producing code
- Example of corrected code
Key insight: Mypy errors often indicate misunderstanding about what types a function accepts or returns.
-
Trace Type Flow
Follow the data flow to understand type relationships:
a. Read the error location:
claudeRead("/path/to/file.py")b. Identify the type mismatch:
- What type does mypy think the variable is?
- What type does mypy expect?
- Where does the variable get its type?
c. Trace upstream:
- Read function signatures
- Check return type annotations
- Review variable assignments
d. Check library type stubs:
- If the error involves a library, check its type stubs
- Use
python -c "import library; print(library.__file__)"to locate - Read
.pyistub files orpy.typedmarker
-
Apply Common Methodology
Follow steps 1-4 in the Common Resolution Methodology section above:
- Load python3-development skill
- Pass through Suppression Gate
- Check Architectural Context
- Verify Resolution
-
Implement Elegant Fix
Choose the appropriate fix strategy:
Strategy A: Fix the type annotation (if annotation is wrong)
python# Before: Function returns dict but annotated as returning Response def get_data() -> Response: return {"key": "value"} # mypy error: Incompatible return value type # After: Correct annotation to match actual return def get_data() -> dict[str, str]: return {"key": "value"}Strategy B: Fix the implementation (if annotation is correct)
python# Before: Function should return Response but returns dict def get_data() -> Response: return {"key": "value"} # mypy error: Incompatible return value type # After: Fix implementation to return correct type def get_data() -> Response: return Response(data={"key": "value"})Strategy C: Add type narrowing (if type is conditional)
python# Before: Mypy can't prove value is not None def process(value: str | None) -> str: return value.upper() # mypy error: Item "None" has no attribute "upper" # After: Add type guard def process(value: str | None) -> str: if value is None: raise ValueError("value cannot be None") return value.upper()Strategy D: Use TypeGuard for complex narrowing
pythonfrom typing import TypeGuard def is_valid_response(data: dict[str, Any]) -> TypeGuard[dict[str, str]]: return all(isinstance(v, str) for v in data.values()) def process(data: dict[str, Any]) -> dict[str, str]: if not is_valid_response(data): raise ValueError("Invalid data format") return data # mypy now knows this is dict[str, str]
Pyright/Basedpyright Resolution Workflow
When to use: Type checking errors with pyright diagnostic rules (reportGeneralTypeIssues, reportOptionalMemberAccess, reportUnknownVariableType, etc.)
Resolution Process:
-
Research the Diagnostic Rule
Pyright errors reference diagnostic rule names like
reportOptionalMemberAccessorreportGeneralTypeIssues.Look up the rule in basedpyright documentation:
For rule settings and descriptions:
Use MCP tools for documentation lookup (in order of preference):
claude# Option 1 (Preferred): Use Ref MCP for high-fidelity documentation mcp__Ref__ref_search_documentation(query="basedpyright {RULE_NAME} diagnostic rule configuration") # Then read the URL from results: mcp__Ref__ref_read_url(url="<exact_url_from_search_results>") # Option 2: Use exa for code context if Ref doesn't have it mcp__exa__get_code_context_exa(query="basedpyright {RULE_NAME} diagnostic rule examples") # Fallback: Use WebFetch only if MCP tools don't work WebFetch(url="https://docs.basedpyright.com/latest/configuration/config-files/", prompt="Find documentation for diagnostic rule {RULE_NAME}")For features and PEP support:
claude# Option 1 (Preferred): Use Ref MCP for high-fidelity documentation mcp__Ref__ref_search_documentation(query="basedpyright Python typing features PEP {RULE_NAME}") mcp__Ref__ref_read_url(url="<exact_url_from_search_results>") # Fallback: Use WebFetch only if MCP tools don't work WebFetch(url="https://docs.basedpyright.com/latest/getting_started/features/", prompt="Explain what Python typing features and PEPs are covered related to {RULE_NAME}")Motivation: Pyright is more strict than mypy in many areas. Understanding what the rule enforces helps identify whether the issue is a genuine type safety problem or overly strict checking.
-
Read Diagnostic Rule Documentation
The basedpyright documentation explains:
- What type safety issue the rule detects
- Configuration levels (basic, standard, strict, all)
- Whether the rule can be disabled per-project
- Related typing features and PEPs
-
Read the Affected Code
Read the complete file containing the type error:
claudeRead("/path/to/file.py")Focus on:
- The exact line with the error
- Type annotations in the surrounding function/class
- Import statements for typing constructs
-
Understand the Type Inference Issue
Pyright has sophisticated type inference. Common issues:
Optional member access:
python# Error: reportOptionalMemberAccess value: str | None = get_value() result = value.upper() # Error: 'value' could be 'None'Unknown variable type:
python# Error: reportUnknownVariableType result = some_function() # some_function has no return type annotationType narrowing not recognized:
python# Error: pyright doesn't recognize the narrowing value: int | str = get_value() if type(value) == int: # Use isinstance() instead result = value + 1 -
Apply Common Methodology
Follow steps 1-4 in the Common Resolution Methodology section above:
- Load python3-development skill
- Pass through Suppression Gate
- Check Architectural Context
- Verify Resolution
-
Implement Elegant Fix
Choose the appropriate fix strategy:
Strategy A: Add type narrowing guards
python# Before: def process(value: str | None) -> str: return value.upper() # reportOptionalMemberAccess # After: def process(value: str | None) -> str: if value is None: raise ValueError("value is required") return value.upper() # pyright knows value is str hereStrategy B: Add missing type annotations
python# Before: def fetch_data(): # reportUnknownVariableType on callers return {"key": "value"} # After: def fetch_data() -> dict[str, str]: return {"key": "value"}Strategy C: Use assert for type narrowing
python# Before: value: int | str = get_value() result = value + 1 # reportGeneralTypeIssues # After: value: int | str = get_value() assert isinstance(value, int), "Expected int" result = value + 1 # pyright knows value is intStrategy D: Use typing.cast for complex cases
pythonfrom typing import cast # Before: data: dict[str, Any] = get_data() name: str = data["name"] # reportUnknownVariableType # After (if you've validated data structure): from typing import TypedDict class UserData(TypedDict): name: str age: int data = cast(UserData, get_data()) name: str = data["name"] # pyright knows this is strWhen all strategies fail: Apply the Suppression Gate — document approaches tried and fundamental constraint, then return UNRESOLVED to the orchestrator. The pyproject.toml severity level is a project configuration decision, not a linting resolution action. Config changes require explicit user approval via the UNRESOLVED escalation path, not autonomous agent action.
Integration: Resolution Process with python3-development
All linter resolution workflows integrate with the python3-development skill at the implementation stage. This integration ensures:
-
Modern Python Patterns: Fixes use Python 3.11+ syntax
- Native generics (
list[str]notList[str]) - Union syntax (
str | NonenotOptional[str]) - Structural pattern matching where appropriate
- Native generics (
-
Idiomatic Code: Solutions follow Python best practices
- Clear naming conventions
- Appropriate use of comprehensions
- Proper exception handling
- Single Responsibility Principle
-
Type Safety: Type annotations are complete and accurate
- Precise return types
- Correct parameter types
- Proper use of generics and protocols
-
Project Consistency: Fixes align with existing codebase patterns
- Consistent with project's CLAUDE.md standards
- Matches existing module organization
- Follows project-specific conventions
Activation pattern:
[Identify linting issue] → [Research rule] → [Read code] → [Check architecture]
→ [Load python3-development skill] → [Implement elegant fix] → [Verify]
Related Skills
- holistic-linting - Core linting skill with linter detection and resource documentation
- holistic-linting-orchestrator - Orchestrator delegation workflows
- Pre-Existing Issues Protocol - Recording and triage pipeline for issues found outside current task scope
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
ccc
This skill should be used when code search is needed (whether explicitly requested or as part of completing a task), when indexing the codebase after changes, or when the user asks about ccc, cocoindex-code, or the codebase index. Trigger phrases include 'search the codebase', 'find code related to', 'update the index', 'ccc', 'cocoindex-code'.
agent-browser
Browser automation CLI for AI agents. Use when the user needs to interact with websites, including navigating pages, filling forms, clicking buttons, taking screenshots, extracting data, testing web apps, or automating any browser task. Triggers include requests to "open a website", "fill out a form", "click a button", "take a screenshot", "scrape data from a page", "test this web app", "login to a site", "automate browser actions", or any task requiring programmatic web interaction.
delegate
Quick delegation template for sub-agent prompts. Use when assigning work to a sub-agent, before invoking the Agent tool, or when preparing prompts for specialized agents. Provides the WHERE-WHAT-WHY framework. For comprehensive delegation guidance, activate the agent-orchestration how-to-delegate skill.
swarm-spawning
Spawn agents and teammates in Claude Code swarms. Use when choosing between subagents vs teammates, selecting agent types (Explore, Plan, general-purpose, plugin agents), configuring spawn backends (in-process, tmux, iterm2), or setting environment variables for spawned agents.
knowledge-explorer
Manage the research/ knowledge base (KB) of tool and library research entries. Use when browsing KB topics, adding new research entries, updating existing entries with dated revisions, fetching GitHub repo metadata into a draft KB entry, or migrating old-format entries to skill-spec frontmatter. Triggers on tasks like "what do we have on X", "add this to the KB", "update the KB entry for Y", "fetch github info for owner/repo", or "migrate old entries".
design-anti-patterns
Enforce anti-AI UI design rules based on the Uncodixfy methodology. Use when generating HTML, CSS, React, Vue, Svelte, or any frontend UI code. Prevents "Codex UI" — the generic AI aesthetic of soft gradients, floating panels, oversized rounded corners, glassmorphism, hero sections in dashboards, and decorative copy. Applies constraints from Linear/Raycast/Stripe/GitHub design philosophy: functional, honest, human-designed interfaces. Triggers on: UI generation, dashboard building, frontend component creation, CSS styling, landing page design, or any task producing visual interface code.
Didn't find tool you were looking for?