PluginBench
Skill
Pass
Audit score 90

python-project-structure

wshobson/agents

Design well-organized Python projects with clear module boundaries, explicit public interfaces, and maintainable directory structures.

What is python-project-structure?

Guidance for organizing Python projects, defining module public APIs with __all__, and planning directory layouts. Use when starting new projects, reorganizing codebases, or establishing module boundaries and architectural patterns.

  • Define explicit public interfaces using __all__ to clarify what's internal vs. external
  • Organize code into focused modules with single responsibilities, splitting files around 300-500 lines
  • Structure directories with flat hierarchies, adding depth only for genuine sub-domains
  • Choose and apply consistent test file placement strategies (colocated or parallel directories)
  • Implement layered or domain-driven architecture patterns for separation of concerns
  • Use absolute imports for clarity and reliability across the project

How to install python-project-structure

npx skills add https://github.com/wshobson/agents --skill python-project-structure
Claude Code
Cursor
Windsurf
Cline

How to use python-project-structure

  1. 1.Assess your project scope: new project, reorganization, or library design
  2. 2.Choose a directory structure pattern (flat, layered, or domain-driven) based on complexity
  3. 3.Define module boundaries by grouping related code that changes together
  4. 4.Create __all__ lists in __init__.py files to explicitly declare public interfaces
  5. 5.Apply consistent naming conventions (snake_case for files, match class names to file names)
  6. 6.Select a test file organization strategy and apply it uniformly
  7. 7.Use absolute imports throughout the project for reliability
  8. 8.Document your structure choices in the project README

Use cases

Good for
  • Starting a new Python project from scratch with clear module organization
  • Reorganizing an existing codebase to improve discoverability and maintainability
  • Defining public APIs for a reusable library package
  • Deciding between flat vs. nested directory structures for a growing application
  • Planning test file placement strategy for a new project
Who it's for
  • Python developers starting new projects
  • Teams reorganizing existing codebases
  • Library and package maintainers
  • Developers designing module boundaries and public interfaces
  • Architects planning application structure

python-project-structure FAQ

When should I split a module into multiple files?

Consider splitting when a file handles multiple unrelated responsibilities, grows beyond 300-500 lines depending on complexity, or contains classes that change for different reasons. Each file should focus on a single concept.

What's the difference between layered and domain-driven architecture?

Layered architecture organizes by technical layer (api, services, repositories, models) with dependencies flowing downward. Domain-driven organizes by business domain (users, orders, products) with each domain containing its own layers. Choose layered for simpler apps, domain-driven for complex multi-domain applications.

Should I use relative or absolute imports?

Use absolute imports for clarity and reliability. Relative imports can break when modules are moved or reorganized. Absolute imports make dependencies explicit and easier to understand.

How do I define a public API for my package?

Use __all__ in your __init__.py to list what's public. Import the items you want to expose, add them to __all__, and anything not listed becomes an internal implementation detail. This makes the public interface explicit for users.

Should tests be colocated with code or in a separate directory?

Both approaches work; choose one and apply it consistently. Colocated tests (test_module.py next to module.py) make coverage gaps obvious. Parallel test directories (tests/ mirroring src/) provide cleaner separation and are standard for larger projects.

Full instructions (SKILL.md)

Source of truth, from wshobson/agents.


name: python-project-structure description: Python project organization, module architecture, and public API design. Use when setting up new projects, organizing modules, defining public interfaces with all, or planning directory layouts.

Python Project Structure & Module Architecture

Design well-organized Python projects with clear module boundaries, explicit public interfaces, and maintainable directory structures. Good organization makes code discoverable and changes predictable.

When to Use This Skill

  • Starting a new Python project from scratch
  • Reorganizing an existing codebase for clarity
  • Defining module public APIs with __all__
  • Deciding between flat and nested directory structures
  • Determining test file placement strategies
  • Creating reusable library packages

Core Concepts

1. Module Cohesion

Group related code that changes together. A module should have a single, clear purpose.

2. Explicit Interfaces

Define what's public with __all__. Everything not listed is an internal implementation detail.

3. Flat Hierarchies

Prefer shallow directory structures. Add depth only for genuine sub-domains.

4. Consistent Conventions

Apply naming and organization patterns uniformly across the project.

Quick Start

myproject/
├── src/
│   └── myproject/
│       ├── __init__.py
│       ├── services/
│       ├── models/
│       └── api/
├── tests/
├── pyproject.toml
└── README.md

Fundamental Patterns

Pattern 1: One Concept Per File

Each file should focus on a single concept or closely related set of functions. Consider splitting when a file:

  • Handles multiple unrelated responsibilities
  • Grows beyond 300-500 lines (varies by complexity)
  • Contains classes that change for different reasons
# Good: Focused files
# user_service.py - User business logic
# user_repository.py - User data access
# user_models.py - User data structures

# Avoid: Kitchen sink files
# user.py - Contains service, repository, models, utilities...

Pattern 2: Explicit Public APIs with __all__

Define the public interface for every module. Unlisted members are internal implementation details.

# mypackage/services/__init__.py
from .user_service import UserService
from .order_service import OrderService
from .exceptions import ServiceError, ValidationError

__all__ = [
    "UserService",
    "OrderService",
    "ServiceError",
    "ValidationError",
]

# Internal helpers remain private by omission
# from .internal_helpers import _validate_input  # Not exported

Pattern 3: Flat Directory Structure

Prefer minimal nesting. Deep hierarchies make imports verbose and navigation difficult.

# Preferred: Flat structure
project/
├── api/
│   ├── routes.py
│   └── middleware.py
├── services/
│   ├── user_service.py
│   └── order_service.py
├── models/
│   ├── user.py
│   └── order.py
└── utils/
    └── validation.py

# Avoid: Deep nesting
project/core/internal/services/impl/user/

Add sub-packages only when there's a genuine sub-domain requiring isolation.

Pattern 4: Test File Organization

Choose one approach and apply it consistently throughout the project.

Option A: Colocated Tests

src/
├── user_service.py
├── test_user_service.py
├── order_service.py
└── test_order_service.py

Benefits: Tests live next to the code they verify. Easy to see coverage gaps.

Option B: Parallel Test Directory

src/
├── services/
│   ├── user_service.py
│   └── order_service.py
tests/
├── services/
│   ├── test_user_service.py
│   └── test_order_service.py

Benefits: Clean separation between production and test code. Standard for larger projects.

Advanced Patterns

Pattern 5: Package Initialization

Use __init__.py to provide a clean public interface for package consumers.

# mypackage/__init__.py
"""MyPackage - A library for doing useful things."""

from .core import MainClass, HelperClass
from .exceptions import PackageError, ConfigError
from .config import Settings

__all__ = [
    "MainClass",
    "HelperClass",
    "PackageError",
    "ConfigError",
    "Settings",
]

__version__ = "1.0.0"

Consumers can then import directly from the package:

from mypackage import MainClass, Settings

Pattern 6: Layered Architecture

Organize code by architectural layer for clear separation of concerns.

myapp/
├── api/           # HTTP handlers, request/response
│   ├── routes/
│   └── middleware/
├── services/      # Business logic
├── repositories/  # Data access
├── models/        # Domain entities
├── schemas/       # API schemas (Pydantic)
└── config/        # Configuration

Each layer should only depend on layers below it, never above.

Pattern 7: Domain-Driven Structure

For complex applications, organize by business domain rather than technical layer.

ecommerce/
├── users/
│   ├── models.py
│   ├── services.py
│   ├── repository.py
│   └── api.py
├── orders/
│   ├── models.py
│   ├── services.py
│   ├── repository.py
│   └── api.py
└── shared/
    ├── database.py
    └── exceptions.py

File and Module Naming

Conventions

  • Use snake_case for all file and module names: user_repository.py
  • Avoid abbreviations that obscure meaning: user_repository.py not usr_repo.py
  • Match class names to file names: UserService in user_service.py

Import Style

Use absolute imports for clarity and reliability:

# Preferred: Absolute imports
from myproject.services import UserService
from myproject.models import User

# Avoid: Relative imports
from ..services import UserService
from . import models

Relative imports can break when modules are moved or reorganized.

Best Practices Summary

  1. Keep files focused - One concept per file, consider splitting at 300-500 lines (varies by complexity)
  2. Define __all__ explicitly - Make public interfaces clear
  3. Prefer flat structures - Add depth only for genuine sub-domains
  4. Use absolute imports - More reliable and clearer
  5. Be consistent - Apply patterns uniformly across the project
  6. Match names to content - File names should describe their purpose
  7. Separate concerns - Keep layers distinct and dependencies flowing one direction
  8. Document your structure - Include a README explaining the organization