@sylix/coworker 2.0.10 → 2.0.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/commands/slash/config.d.ts.map +1 -1
- package/dist/commands/slash/config.js +23 -5
- package/dist/commands/slash/config.js.map +1 -1
- package/dist/commands/slash/todo.js +1 -1
- package/dist/commands/slash/todo.js.map +1 -1
- package/dist/core/CoWorkerAgent.d.ts.map +1 -1
- package/dist/core/CoWorkerAgent.js +6 -3
- package/dist/core/CoWorkerAgent.js.map +1 -1
- package/dist/permissions/PermissionInterceptor.js +1 -1
- package/dist/permissions/PermissionInterceptor.js.map +1 -1
- package/dist/skills/defaults/accessibility/screen-reader-testing.md +545 -0
- package/dist/skills/defaults/accessibility/wcag-audit-patterns.md +555 -0
- package/dist/skills/defaults/ai-ml/rag.md +276 -0
- package/dist/skills/defaults/backend-development/api-design-principles.md +528 -0
- package/dist/skills/defaults/backend-development/api-design.md +285 -0
- package/dist/skills/defaults/backend-development/architecture-patterns.md +494 -0
- package/dist/skills/defaults/backend-development/async-python.md +237 -0
- package/dist/skills/defaults/backend-development/auth-implementation-patterns.md +638 -0
- package/dist/skills/defaults/backend-development/bazel-build-optimization.md +387 -0
- package/dist/skills/defaults/backend-development/billing-automation/SKILL.md +566 -0
- package/dist/skills/defaults/backend-development/code-review-excellence.md +538 -0
- package/dist/skills/defaults/backend-development/cqrs-implementation.md +554 -0
- package/dist/skills/defaults/backend-development/database-design.md +305 -0
- package/dist/skills/defaults/backend-development/debugging-strategies.md +536 -0
- package/dist/skills/defaults/backend-development/e2e-testing-patterns.md +544 -0
- package/dist/skills/defaults/backend-development/error-handling-patterns.md +641 -0
- package/dist/skills/defaults/backend-development/fastapi-templates.md +559 -0
- package/dist/skills/defaults/backend-development/fastapi.md +309 -0
- package/dist/skills/defaults/backend-development/git-advanced-workflows.md +405 -0
- package/dist/skills/defaults/backend-development/microservices-patterns.md +595 -0
- package/dist/skills/defaults/backend-development/microservices.md +284 -0
- package/dist/skills/defaults/backend-development/monorepo-management.md +623 -0
- package/dist/skills/defaults/backend-development/nodejs-backend-patterns.md +1048 -0
- package/dist/skills/defaults/backend-development/nx-workspace-patterns.md +457 -0
- package/dist/skills/defaults/backend-development/paypal-integration/SKILL.md +478 -0
- package/dist/skills/defaults/backend-development/pci-compliance/SKILL.md +480 -0
- package/dist/skills/defaults/backend-development/python-anti-patterns.md +349 -0
- package/dist/skills/defaults/backend-development/python-background-jobs.md +364 -0
- package/dist/skills/defaults/backend-development/python-code-style.md +360 -0
- package/dist/skills/defaults/backend-development/python-configuration.md +368 -0
- package/dist/skills/defaults/backend-development/python-design-patterns.md +296 -0
- package/dist/skills/defaults/backend-development/python-error-handling.md +323 -0
- package/dist/skills/defaults/backend-development/python-packaging.md +887 -0
- package/dist/skills/defaults/backend-development/python-performance-optimization.md +874 -0
- package/dist/skills/defaults/backend-development/python-project-structure.md +252 -0
- package/dist/skills/defaults/backend-development/python-resilience.md +376 -0
- package/dist/skills/defaults/backend-development/python-resource-management.md +421 -0
- package/dist/skills/defaults/backend-development/python-type-safety.md +428 -0
- package/dist/skills/defaults/backend-development/sql-optimization-patterns.md +509 -0
- package/dist/skills/defaults/backend-development/stripe-integration/SKILL.md +522 -0
- package/dist/skills/defaults/backend-development/turborepo-caching.md +376 -0
- package/dist/skills/defaults/blockchain/defi-protocol-templates.md +430 -0
- package/dist/skills/defaults/blockchain/nft-standards.md +364 -0
- package/dist/skills/defaults/blockchain/solidity-security.md +514 -0
- package/dist/skills/defaults/blockchain/web3-testing.md +360 -0
- package/dist/skills/defaults/business/competitive-landscape/SKILL.md +527 -0
- package/dist/skills/defaults/business/market-sizing-analysis/SKILL.md +451 -0
- package/dist/skills/defaults/business/startup-financial-modeling/SKILL.md +494 -0
- package/dist/skills/defaults/business/startup-metrics-framework/SKILL.md +564 -0
- package/dist/skills/defaults/business/team-composition-analysis.md +437 -0
- package/dist/skills/defaults/compliance/employment-contract-templates/SKILL.md +527 -0
- package/dist/skills/defaults/compliance/gdpr-data-handling/SKILL.md +630 -0
- package/dist/skills/defaults/data-engineering/airflow-dag-patterns.md +436 -0
- package/dist/skills/defaults/data-engineering/airflow.md +519 -0
- package/dist/skills/defaults/data-engineering/data-quality.md +583 -0
- package/dist/skills/defaults/data-engineering/dbt-transformation-patterns.md +482 -0
- package/dist/skills/defaults/data-engineering/dbt.md +556 -0
- package/dist/skills/defaults/data-engineering/ml-pipeline-workflow/SKILL.md +247 -0
- package/dist/skills/defaults/data-engineering/spark-optimization.md +348 -0
- package/dist/skills/defaults/data-engineering/spark.md +411 -0
- package/dist/skills/defaults/database/postgresql.md +202 -0
- package/dist/skills/defaults/debugging/systematic-debugging.md +249 -0
- package/dist/skills/defaults/devops/architecture-decision-records.md +448 -0
- package/dist/skills/defaults/devops/changelog-automation.md +580 -0
- package/dist/skills/defaults/devops/cicd.md +314 -0
- package/dist/skills/defaults/devops/cloud.md +263 -0
- package/dist/skills/defaults/devops/code-review-excellence.md +299 -0
- package/dist/skills/defaults/devops/cost-optimization.md +295 -0
- package/dist/skills/defaults/devops/deployment-pipeline-design.md +356 -0
- package/dist/skills/defaults/devops/docker.md +281 -0
- package/dist/skills/defaults/devops/git-workflows.md +205 -0
- package/dist/skills/defaults/devops/github-actions.md +311 -0
- package/dist/skills/defaults/devops/gitlab-ci-patterns.md +266 -0
- package/dist/skills/defaults/devops/hybrid-cloud-networking.md +241 -0
- package/dist/skills/defaults/devops/istio-traffic-management.md +327 -0
- package/dist/skills/defaults/devops/kubernetes.md +339 -0
- package/dist/skills/defaults/devops/linkerd-patterns.md +311 -0
- package/dist/skills/defaults/devops/multi-cloud-architecture.md +181 -0
- package/dist/skills/defaults/devops/observability.md +243 -0
- package/dist/skills/defaults/devops/openapi-spec-generation.md +1024 -0
- package/dist/skills/defaults/devops/postmortem-writing.md +396 -0
- package/dist/skills/defaults/devops/prometheus-configuration.md +265 -0
- package/dist/skills/defaults/devops/secrets-management.md +341 -0
- package/dist/skills/defaults/devops/service-mesh-observability.md +385 -0
- package/dist/skills/defaults/devops/terraform-module-library.md +244 -0
- package/dist/skills/defaults/finance/backtesting-frameworks/SKILL.md +663 -0
- package/dist/skills/defaults/finance/risk-metrics-calculation/SKILL.md +557 -0
- package/dist/skills/defaults/frontend/accessibility-compliance.md +420 -0
- package/dist/skills/defaults/frontend/design-system-patterns.md +337 -0
- package/dist/skills/defaults/frontend/interaction-design.md +327 -0
- package/dist/skills/defaults/frontend/javascript.md +311 -0
- package/dist/skills/defaults/frontend/modern-javascript-patterns.md +927 -0
- package/dist/skills/defaults/frontend/react-native-design.md +440 -0
- package/dist/skills/defaults/frontend/react.md +345 -0
- package/dist/skills/defaults/frontend/responsive-design.md +472 -0
- package/dist/skills/defaults/frontend/tailwind-design-system.md +337 -0
- package/dist/skills/defaults/frontend/typescript-advanced-types.md +724 -0
- package/dist/skills/defaults/frontend/typescript.md +334 -0
- package/dist/skills/defaults/frontend/visual-design-foundations.md +326 -0
- package/dist/skills/defaults/frontend/web-component-design.md +279 -0
- package/dist/skills/defaults/game-development/godot-gdscript-patterns.md +188 -0
- package/dist/skills/defaults/game-development/unity-ecs-patterns.md +594 -0
- package/dist/skills/defaults/kubernetes/gitops-workflow.md +285 -0
- package/dist/skills/defaults/kubernetes/gitops.md +280 -0
- package/dist/skills/defaults/kubernetes/helm-chart-scaffolding.md +553 -0
- package/dist/skills/defaults/kubernetes/helm.md +343 -0
- package/dist/skills/defaults/kubernetes/k8s-manifest-generator.md +501 -0
- package/dist/skills/defaults/kubernetes/k8s-security-policies.md +342 -0
- package/dist/skills/defaults/kubernetes/manifests.md +330 -0
- package/dist/skills/defaults/kubernetes/security.md +337 -0
- package/dist/skills/defaults/llm-application/embedding-strategies.md +608 -0
- package/dist/skills/defaults/llm-application/hybrid-search-implementation.md +570 -0
- package/dist/skills/defaults/llm-application/hybrid-search.md +570 -0
- package/dist/skills/defaults/llm-application/langchain-architecture.md +666 -0
- package/dist/skills/defaults/llm-application/langchain.md +259 -0
- package/dist/skills/defaults/llm-application/llm-evaluation.md +695 -0
- package/dist/skills/defaults/llm-application/prompt-engineering-patterns.md +449 -0
- package/dist/skills/defaults/llm-application/prompt-engineering.md +219 -0
- package/dist/skills/defaults/llm-application/rag-implementation.md +434 -0
- package/dist/skills/defaults/llm-application/similarity-search-patterns.md +560 -0
- package/dist/skills/defaults/llm-application/similarity-search.md +560 -0
- package/dist/skills/defaults/llm-application/vector-index-tuning.md +523 -0
- package/dist/skills/defaults/mobile/mobile-android-design.md +440 -0
- package/dist/skills/defaults/mobile/mobile-ios-design.md +266 -0
- package/dist/skills/defaults/monitoring/distributed-tracing.md +436 -0
- package/dist/skills/defaults/monitoring/grafana-dashboards.md +370 -0
- package/dist/skills/defaults/monitoring/prometheus-configuration.md +379 -0
- package/dist/skills/defaults/monitoring/slo-implementation.md +323 -0
- package/dist/skills/defaults/refactoring/code-refactoring.md +349 -0
- package/dist/skills/defaults/security/anti-reversing-techniques/SKILL.md +559 -0
- package/dist/skills/defaults/security/auditor.md +168 -0
- package/dist/skills/defaults/security/binary-analysis-patterns/SKILL.md +438 -0
- package/dist/skills/defaults/security/memory-forensics/SKILL.md +483 -0
- package/dist/skills/defaults/security/mtls-configuration.md +349 -0
- package/dist/skills/defaults/security/protocol-reverse-engineering/SKILL.md +520 -0
- package/dist/skills/defaults/security/sast-configuration.md +182 -0
- package/dist/skills/defaults/security/security.md +313 -0
- package/dist/skills/defaults/security/stride-analysis.md +273 -0
- package/dist/skills/defaults/security/threat-mitigation-mapping.md +290 -0
- package/dist/skills/defaults/systems/bash-defensive-patterns/SKILL.md +539 -0
- package/dist/skills/defaults/systems/bats-testing-patterns/SKILL.md +631 -0
- package/dist/skills/defaults/systems/go-concurrency-patterns.md +657 -0
- package/dist/skills/defaults/systems/memory-safety-patterns.md +605 -0
- package/dist/skills/defaults/systems/rust-async-patterns.md +519 -0
- package/dist/skills/defaults/systems/shellcheck-configuration/SKILL.md +456 -0
- package/dist/skills/defaults/team-collaboration/multi-reviewer-patterns.md +126 -0
- package/dist/skills/defaults/team-collaboration/parallel-feature-development.md +151 -0
- package/dist/skills/defaults/testing/javascript-testing-patterns.md +1021 -0
- package/dist/skills/defaults/testing/python-testing-patterns.md +351 -0
- package/dist/skills/defaults/testing/testing.md +332 -0
- package/dist/skills/defaults/workflows/context-driven-development.md +384 -0
- package/dist/skills/defaults/workflows/track-management.md +592 -0
- package/dist/skills/defaults/workflows/workflow-patterns.md +622 -0
- package/dist/skills/index.d.ts +11 -0
- package/dist/skills/index.d.ts.map +1 -0
- package/dist/skills/index.js +129 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/utils/character.js +6 -9
- package/dist/utils/character.js.map +1 -1
- package/dist/utils/contextManager.js +3 -7
- package/dist/utils/contextManager.js.map +1 -1
- package/dist/utils/inputbar.d.ts.map +1 -1
- package/dist/utils/inputbar.js +8 -1
- package/dist/utils/inputbar.js.map +1 -1
- package/dist/utils/output.d.ts.map +1 -1
- package/dist/utils/output.js +3 -35
- package/dist/utils/output.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,428 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: python-type-safety
|
|
3
|
+
description: Python type safety with type hints, generics, protocols, and strict type checking. Use when adding type annotations, implementing generic classes, defining structural interfaces, or configuring mypy/pyright.
|
|
4
|
+
---
|
|
5
|
+
|
|
6
|
+
# Python Type Safety
|
|
7
|
+
|
|
8
|
+
Leverage Python's type system to catch errors at static analysis time. Type annotations serve as enforced documentation that tooling validates automatically.
|
|
9
|
+
|
|
10
|
+
## When to Use This Skill
|
|
11
|
+
|
|
12
|
+
- Adding type hints to existing code
|
|
13
|
+
- Creating generic, reusable classes
|
|
14
|
+
- Defining structural interfaces with protocols
|
|
15
|
+
- Configuring mypy or pyright for strict checking
|
|
16
|
+
- Understanding type narrowing and guards
|
|
17
|
+
- Building type-safe APIs and libraries
|
|
18
|
+
|
|
19
|
+
## Core Concepts
|
|
20
|
+
|
|
21
|
+
### 1. Type Annotations
|
|
22
|
+
|
|
23
|
+
Declare expected types for function parameters, return values, and variables.
|
|
24
|
+
|
|
25
|
+
### 2. Generics
|
|
26
|
+
|
|
27
|
+
Write reusable code that preserves type information across different types.
|
|
28
|
+
|
|
29
|
+
### 3. Protocols
|
|
30
|
+
|
|
31
|
+
Define structural interfaces without inheritance (duck typing with type safety).
|
|
32
|
+
|
|
33
|
+
### 4. Type Narrowing
|
|
34
|
+
|
|
35
|
+
Use guards and conditionals to narrow types within code blocks.
|
|
36
|
+
|
|
37
|
+
## Quick Start
|
|
38
|
+
|
|
39
|
+
```python
|
|
40
|
+
def get_user(user_id: str) -> User | None:
|
|
41
|
+
"""Return type makes 'might not exist' explicit."""
|
|
42
|
+
...
|
|
43
|
+
|
|
44
|
+
# Type checker enforces handling None case
|
|
45
|
+
user = get_user("123")
|
|
46
|
+
if user is None:
|
|
47
|
+
raise UserNotFoundError("123")
|
|
48
|
+
print(user.name) # Type checker knows user is User here
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
## Fundamental Patterns
|
|
52
|
+
|
|
53
|
+
### Pattern 1: Annotate All Public Signatures
|
|
54
|
+
|
|
55
|
+
Every public function, method, and class should have type annotations.
|
|
56
|
+
|
|
57
|
+
```python
|
|
58
|
+
def get_user(user_id: str) -> User:
|
|
59
|
+
"""Retrieve user by ID."""
|
|
60
|
+
...
|
|
61
|
+
|
|
62
|
+
def process_batch(
|
|
63
|
+
items: list[Item],
|
|
64
|
+
max_workers: int = 4,
|
|
65
|
+
) -> BatchResult[ProcessedItem]:
|
|
66
|
+
"""Process items concurrently."""
|
|
67
|
+
...
|
|
68
|
+
|
|
69
|
+
class UserRepository:
|
|
70
|
+
def __init__(self, db: Database) -> None:
|
|
71
|
+
self._db = db
|
|
72
|
+
|
|
73
|
+
async def find_by_id(self, user_id: str) -> User | None:
|
|
74
|
+
"""Return User if found, None otherwise."""
|
|
75
|
+
...
|
|
76
|
+
|
|
77
|
+
async def find_by_email(self, email: str) -> User | None:
|
|
78
|
+
...
|
|
79
|
+
|
|
80
|
+
async def save(self, user: User) -> User:
|
|
81
|
+
"""Save and return user with generated ID."""
|
|
82
|
+
...
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
Use `mypy --strict` or `pyright` in CI to catch type errors early. For existing projects, enable strict mode incrementally using per-module overrides.
|
|
86
|
+
|
|
87
|
+
### Pattern 2: Use Modern Union Syntax
|
|
88
|
+
|
|
89
|
+
Python 3.10+ provides cleaner union syntax.
|
|
90
|
+
|
|
91
|
+
```python
|
|
92
|
+
# Preferred (3.10+)
|
|
93
|
+
def find_user(user_id: str) -> User | None:
|
|
94
|
+
...
|
|
95
|
+
|
|
96
|
+
def parse_value(v: str) -> int | float | str:
|
|
97
|
+
...
|
|
98
|
+
|
|
99
|
+
# Older style (still valid, needed for 3.9)
|
|
100
|
+
from typing import Optional, Union
|
|
101
|
+
|
|
102
|
+
def find_user(user_id: str) -> Optional[User]:
|
|
103
|
+
...
|
|
104
|
+
```
|
|
105
|
+
|
|
106
|
+
### Pattern 3: Type Narrowing with Guards
|
|
107
|
+
|
|
108
|
+
Use conditionals to narrow types for the type checker.
|
|
109
|
+
|
|
110
|
+
```python
|
|
111
|
+
def process_user(user_id: str) -> UserData:
|
|
112
|
+
user = find_user(user_id)
|
|
113
|
+
|
|
114
|
+
if user is None:
|
|
115
|
+
raise UserNotFoundError(f"User {user_id} not found")
|
|
116
|
+
|
|
117
|
+
# Type checker knows user is User here, not User | None
|
|
118
|
+
return UserData(
|
|
119
|
+
name=user.name,
|
|
120
|
+
email=user.email,
|
|
121
|
+
)
|
|
122
|
+
|
|
123
|
+
def process_items(items: list[Item | None]) -> list[ProcessedItem]:
|
|
124
|
+
# Filter and narrow types
|
|
125
|
+
valid_items = [item for item in items if item is not None]
|
|
126
|
+
# valid_items is now list[Item]
|
|
127
|
+
return [process(item) for item in valid_items]
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
### Pattern 4: Generic Classes
|
|
131
|
+
|
|
132
|
+
Create type-safe reusable containers.
|
|
133
|
+
|
|
134
|
+
```python
|
|
135
|
+
from typing import TypeVar, Generic
|
|
136
|
+
|
|
137
|
+
T = TypeVar("T")
|
|
138
|
+
E = TypeVar("E", bound=Exception)
|
|
139
|
+
|
|
140
|
+
class Result(Generic[T, E]):
|
|
141
|
+
"""Represents either a success value or an error."""
|
|
142
|
+
|
|
143
|
+
def __init__(
|
|
144
|
+
self,
|
|
145
|
+
value: T | None = None,
|
|
146
|
+
error: E | None = None,
|
|
147
|
+
) -> None:
|
|
148
|
+
if (value is None) == (error is None):
|
|
149
|
+
raise ValueError("Exactly one of value or error must be set")
|
|
150
|
+
self._value = value
|
|
151
|
+
self._error = error
|
|
152
|
+
|
|
153
|
+
@property
|
|
154
|
+
def is_success(self) -> bool:
|
|
155
|
+
return self._error is None
|
|
156
|
+
|
|
157
|
+
@property
|
|
158
|
+
def is_failure(self) -> bool:
|
|
159
|
+
return self._error is not None
|
|
160
|
+
|
|
161
|
+
def unwrap(self) -> T:
|
|
162
|
+
"""Get value or raise the error."""
|
|
163
|
+
if self._error is not None:
|
|
164
|
+
raise self._error
|
|
165
|
+
return self._value # type: ignore[return-value]
|
|
166
|
+
|
|
167
|
+
def unwrap_or(self, default: T) -> T:
|
|
168
|
+
"""Get value or return default."""
|
|
169
|
+
if self._error is not None:
|
|
170
|
+
return default
|
|
171
|
+
return self._value # type: ignore[return-value]
|
|
172
|
+
|
|
173
|
+
# Usage preserves types
|
|
174
|
+
def parse_config(path: str) -> Result[Config, ConfigError]:
|
|
175
|
+
try:
|
|
176
|
+
return Result(value=Config.from_file(path))
|
|
177
|
+
except ConfigError as e:
|
|
178
|
+
return Result(error=e)
|
|
179
|
+
|
|
180
|
+
result = parse_config("config.yaml")
|
|
181
|
+
if result.is_success:
|
|
182
|
+
config = result.unwrap() # Type: Config
|
|
183
|
+
```
|
|
184
|
+
|
|
185
|
+
## Advanced Patterns
|
|
186
|
+
|
|
187
|
+
### Pattern 5: Generic Repository
|
|
188
|
+
|
|
189
|
+
Create type-safe data access patterns.
|
|
190
|
+
|
|
191
|
+
```python
|
|
192
|
+
from typing import TypeVar, Generic
|
|
193
|
+
from abc import ABC, abstractmethod
|
|
194
|
+
|
|
195
|
+
T = TypeVar("T")
|
|
196
|
+
ID = TypeVar("ID")
|
|
197
|
+
|
|
198
|
+
class Repository(ABC, Generic[T, ID]):
|
|
199
|
+
"""Generic repository interface."""
|
|
200
|
+
|
|
201
|
+
@abstractmethod
|
|
202
|
+
async def get(self, id: ID) -> T | None:
|
|
203
|
+
"""Get entity by ID."""
|
|
204
|
+
...
|
|
205
|
+
|
|
206
|
+
@abstractmethod
|
|
207
|
+
async def save(self, entity: T) -> T:
|
|
208
|
+
"""Save and return entity."""
|
|
209
|
+
...
|
|
210
|
+
|
|
211
|
+
@abstractmethod
|
|
212
|
+
async def delete(self, id: ID) -> bool:
|
|
213
|
+
"""Delete entity, return True if existed."""
|
|
214
|
+
...
|
|
215
|
+
|
|
216
|
+
class UserRepository(Repository[User, str]):
|
|
217
|
+
"""Concrete repository for Users with string IDs."""
|
|
218
|
+
|
|
219
|
+
async def get(self, id: str) -> User | None:
|
|
220
|
+
row = await self._db.fetchrow(
|
|
221
|
+
"SELECT * FROM users WHERE id = $1", id
|
|
222
|
+
)
|
|
223
|
+
return User(**row) if row else None
|
|
224
|
+
|
|
225
|
+
async def save(self, entity: User) -> User:
|
|
226
|
+
...
|
|
227
|
+
|
|
228
|
+
async def delete(self, id: str) -> bool:
|
|
229
|
+
...
|
|
230
|
+
```
|
|
231
|
+
|
|
232
|
+
### Pattern 6: TypeVar with Bounds
|
|
233
|
+
|
|
234
|
+
Restrict generic parameters to specific types.
|
|
235
|
+
|
|
236
|
+
```python
|
|
237
|
+
from typing import TypeVar
|
|
238
|
+
from pydantic import BaseModel
|
|
239
|
+
|
|
240
|
+
ModelT = TypeVar("ModelT", bound=BaseModel)
|
|
241
|
+
|
|
242
|
+
def validate_and_create(model_cls: type[ModelT], data: dict) -> ModelT:
|
|
243
|
+
"""Create a validated Pydantic model from dict."""
|
|
244
|
+
return model_cls.model_validate(data)
|
|
245
|
+
|
|
246
|
+
# Works with any BaseModel subclass
|
|
247
|
+
class User(BaseModel):
|
|
248
|
+
name: str
|
|
249
|
+
email: str
|
|
250
|
+
|
|
251
|
+
user = validate_and_create(User, {"name": "Alice", "email": "a@b.com"})
|
|
252
|
+
# user is typed as User
|
|
253
|
+
|
|
254
|
+
# Type error: str is not a BaseModel subclass
|
|
255
|
+
result = validate_and_create(str, {"name": "Alice"}) # Error!
|
|
256
|
+
```
|
|
257
|
+
|
|
258
|
+
### Pattern 7: Protocols for Structural Typing
|
|
259
|
+
|
|
260
|
+
Define interfaces without requiring inheritance.
|
|
261
|
+
|
|
262
|
+
```python
|
|
263
|
+
from typing import Protocol, runtime_checkable
|
|
264
|
+
|
|
265
|
+
@runtime_checkable
|
|
266
|
+
class Serializable(Protocol):
|
|
267
|
+
"""Any class that can be serialized to/from dict."""
|
|
268
|
+
|
|
269
|
+
def to_dict(self) -> dict:
|
|
270
|
+
...
|
|
271
|
+
|
|
272
|
+
@classmethod
|
|
273
|
+
def from_dict(cls, data: dict) -> "Serializable":
|
|
274
|
+
...
|
|
275
|
+
|
|
276
|
+
# User satisfies Serializable without inheriting from it
|
|
277
|
+
class User:
|
|
278
|
+
def __init__(self, id: str, name: str) -> None:
|
|
279
|
+
self.id = id
|
|
280
|
+
self.name = name
|
|
281
|
+
|
|
282
|
+
def to_dict(self) -> dict:
|
|
283
|
+
return {"id": self.id, "name": self.name}
|
|
284
|
+
|
|
285
|
+
@classmethod
|
|
286
|
+
def from_dict(cls, data: dict) -> "User":
|
|
287
|
+
return cls(id=data["id"], name=data["name"])
|
|
288
|
+
|
|
289
|
+
def serialize(obj: Serializable) -> str:
|
|
290
|
+
"""Works with any Serializable object."""
|
|
291
|
+
return json.dumps(obj.to_dict())
|
|
292
|
+
|
|
293
|
+
# Works - User matches the protocol
|
|
294
|
+
serialize(User("1", "Alice"))
|
|
295
|
+
|
|
296
|
+
# Runtime checking with @runtime_checkable
|
|
297
|
+
isinstance(User("1", "Alice"), Serializable) # True
|
|
298
|
+
```
|
|
299
|
+
|
|
300
|
+
### Pattern 8: Common Protocol Patterns
|
|
301
|
+
|
|
302
|
+
Define reusable structural interfaces.
|
|
303
|
+
|
|
304
|
+
```python
|
|
305
|
+
from typing import Protocol
|
|
306
|
+
|
|
307
|
+
class Closeable(Protocol):
|
|
308
|
+
"""Resource that can be closed."""
|
|
309
|
+
def close(self) -> None: ...
|
|
310
|
+
|
|
311
|
+
class AsyncCloseable(Protocol):
|
|
312
|
+
"""Async resource that can be closed."""
|
|
313
|
+
async def close(self) -> None: ...
|
|
314
|
+
|
|
315
|
+
class Readable(Protocol):
|
|
316
|
+
"""Object that can be read from."""
|
|
317
|
+
def read(self, n: int = -1) -> bytes: ...
|
|
318
|
+
|
|
319
|
+
class HasId(Protocol):
|
|
320
|
+
"""Object with an ID property."""
|
|
321
|
+
@property
|
|
322
|
+
def id(self) -> str: ...
|
|
323
|
+
|
|
324
|
+
class Comparable(Protocol):
|
|
325
|
+
"""Object that supports comparison."""
|
|
326
|
+
def __lt__(self, other: "Comparable") -> bool: ...
|
|
327
|
+
def __le__(self, other: "Comparable") -> bool: ...
|
|
328
|
+
```
|
|
329
|
+
|
|
330
|
+
### Pattern 9: Type Aliases
|
|
331
|
+
|
|
332
|
+
Create meaningful type names.
|
|
333
|
+
|
|
334
|
+
**Note:** The `type` statement was introduced in Python 3.10 for simple aliases. Generic type statements require Python 3.12+.
|
|
335
|
+
|
|
336
|
+
```python
|
|
337
|
+
# Python 3.10+ type statement for simple aliases
|
|
338
|
+
type UserId = str
|
|
339
|
+
type UserDict = dict[str, Any]
|
|
340
|
+
|
|
341
|
+
# Python 3.12+ type statement with generics
|
|
342
|
+
type Handler[T] = Callable[[Request], T]
|
|
343
|
+
type AsyncHandler[T] = Callable[[Request], Awaitable[T]]
|
|
344
|
+
|
|
345
|
+
# Python 3.9-3.11 style (needed for broader compatibility)
|
|
346
|
+
from typing import TypeAlias
|
|
347
|
+
from collections.abc import Callable, Awaitable
|
|
348
|
+
|
|
349
|
+
UserId: TypeAlias = str
|
|
350
|
+
Handler: TypeAlias = Callable[[Request], Response]
|
|
351
|
+
|
|
352
|
+
# Usage
|
|
353
|
+
def register_handler(path: str, handler: Handler[Response]) -> None:
|
|
354
|
+
...
|
|
355
|
+
```
|
|
356
|
+
|
|
357
|
+
### Pattern 10: Callable Types
|
|
358
|
+
|
|
359
|
+
Type function parameters and callbacks.
|
|
360
|
+
|
|
361
|
+
```python
|
|
362
|
+
from collections.abc import Callable, Awaitable
|
|
363
|
+
|
|
364
|
+
# Sync callback
|
|
365
|
+
ProgressCallback = Callable[[int, int], None] # (current, total)
|
|
366
|
+
|
|
367
|
+
# Async callback
|
|
368
|
+
AsyncHandler = Callable[[Request], Awaitable[Response]]
|
|
369
|
+
|
|
370
|
+
# With named parameters (using Protocol)
|
|
371
|
+
class OnProgress(Protocol):
|
|
372
|
+
def __call__(
|
|
373
|
+
self,
|
|
374
|
+
current: int,
|
|
375
|
+
total: int,
|
|
376
|
+
*,
|
|
377
|
+
message: str = "",
|
|
378
|
+
) -> None: ...
|
|
379
|
+
|
|
380
|
+
def process_items(
|
|
381
|
+
items: list[Item],
|
|
382
|
+
on_progress: ProgressCallback | None = None,
|
|
383
|
+
) -> list[Result]:
|
|
384
|
+
for i, item in enumerate(items):
|
|
385
|
+
if on_progress:
|
|
386
|
+
on_progress(i, len(items))
|
|
387
|
+
...
|
|
388
|
+
```
|
|
389
|
+
|
|
390
|
+
## Configuration
|
|
391
|
+
|
|
392
|
+
### Strict Mode Checklist
|
|
393
|
+
|
|
394
|
+
For `mypy --strict` compliance:
|
|
395
|
+
|
|
396
|
+
```toml
|
|
397
|
+
# pyproject.toml
|
|
398
|
+
[tool.mypy]
|
|
399
|
+
python_version = "3.12"
|
|
400
|
+
strict = true
|
|
401
|
+
warn_return_any = true
|
|
402
|
+
warn_unused_ignores = true
|
|
403
|
+
disallow_untyped_defs = true
|
|
404
|
+
disallow_incomplete_defs = true
|
|
405
|
+
no_implicit_optional = true
|
|
406
|
+
```
|
|
407
|
+
|
|
408
|
+
Incremental adoption goals:
|
|
409
|
+
- All function parameters annotated
|
|
410
|
+
- All return types annotated
|
|
411
|
+
- Class attributes annotated
|
|
412
|
+
- Minimize `Any` usage (acceptable for truly dynamic data)
|
|
413
|
+
- Generic collections use type parameters (`list[str]` not `list`)
|
|
414
|
+
|
|
415
|
+
For existing codebases, enable strict mode per-module using `# mypy: strict` or configure per-module overrides in `pyproject.toml`.
|
|
416
|
+
|
|
417
|
+
## Best Practices Summary
|
|
418
|
+
|
|
419
|
+
1. **Annotate all public APIs** - Functions, methods, class attributes
|
|
420
|
+
2. **Use `T | None`** - Modern union syntax over `Optional[T]`
|
|
421
|
+
3. **Run strict type checking** - `mypy --strict` in CI
|
|
422
|
+
4. **Use generics** - Preserve type info in reusable code
|
|
423
|
+
5. **Define protocols** - Structural typing for interfaces
|
|
424
|
+
6. **Narrow types** - Use guards to help the type checker
|
|
425
|
+
7. **Bound type vars** - Restrict generics to meaningful types
|
|
426
|
+
8. **Create type aliases** - Meaningful names for complex types
|
|
427
|
+
9. **Minimize `Any`** - Use specific types or generics. `Any` is acceptable for truly dynamic data or when interfacing with untyped third-party code
|
|
428
|
+
10. **Document with types** - Types are enforceable documentation
|