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-structureHow to use python-project-structure
- 1.Assess your project scope: new project, reorganization, or library design
- 2.Choose a directory structure pattern (flat, layered, or domain-driven) based on complexity
- 3.Define module boundaries by grouping related code that changes together
- 4.Create __all__ lists in __init__.py files to explicitly declare public interfaces
- 5.Apply consistent naming conventions (snake_case for files, match class names to file names)
- 6.Select a test file organization strategy and apply it uniformly
- 7.Use absolute imports throughout the project for reliability
- 8.Document your structure choices in the project README
Use cases
- 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
- 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
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.
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.
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.
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.
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_casefor all file and module names:user_repository.py - Avoid abbreviations that obscure meaning:
user_repository.pynotusr_repo.py - Match class names to file names:
UserServiceinuser_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
- Keep files focused - One concept per file, consider splitting at 300-500 lines (varies by complexity)
- Define
__all__explicitly - Make public interfaces clear - Prefer flat structures - Add depth only for genuine sub-domains
- Use absolute imports - More reliable and clearer
- Be consistent - Apply patterns uniformly across the project
- Match names to content - File names should describe their purpose
- Separate concerns - Keep layers distinct and dependencies flowing one direction
- Document your structure - Include a README explaining the organization
Related skills
More from wshobson/agents and the wider catalog.
tailwind-design-system
Build production-ready design systems with Tailwind CSS v4, design tokens, and component libraries.
typescript-advanced-types
Master TypeScript's advanced type system: generics, conditional types, mapped types, and utility types for type-safe applications.
nodejs-backend-patterns
Build production-ready Node.js backends with Express/Fastify, middleware patterns, auth, and database integration.
python-performance-optimization
Profile and optimize Python code using cProfile, memory profilers, and performance best practices.
brand-landingpage
Brand-first landing page designer with guided interviews and Stitch-powered iteration.
python-testing-patterns
Implement comprehensive testing strategies with pytest, fixtures, mocking, and test-driven development.