params-proto 3.2.0__tar.gz → 3.2.1__tar.gz
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.
- {params_proto-3.2.0 → params_proto-3.2.1}/.claude-plugin/plugin.json +1 -1
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/workspace.xml +2 -8
- {params_proto-3.2.0 → params_proto-3.2.1}/PKG-INFO +1 -1
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/union_types.md +14 -18
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/release_notes.md +16 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/pyproject.toml +1 -1
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/SKILL.md +1 -1
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto.skill +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/cli/cli_parse.py +70 -1
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_nested_cli.py +138 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.claude-plugin/marketplace.json +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.editorconfig +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.gitignore +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/.gitignore +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/codeStyles/codeStyleConfig.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/inspectionProfiles/Project_Default.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/inspectionProfiles/profiles_settings.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/markdown.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/misc.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/modules.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/params-proto.iml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/ruff.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.idea/vcs.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.readthedocs.yaml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.run/pytest for test_neo_proto_cli.run.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/.run/pytest in test_params_proto.run.xml +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/ANSI_HELP_CONSIDERATIONS.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/CLAUDE.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/LICENSE.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/Makefile +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/README +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/README.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/Makefile +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/api/hyper.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/api/proto.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/api/utils.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/advanced_features.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/basic_usage.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/environment_variables.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/hyperparameter_sweeps.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/index.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/nested_configs.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/quick_start.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_static/ansi.css +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/_static/custom.css +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/api/index.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/api/proto.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/conf.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/examples/basic_usage.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/examples/cli_applications.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/examples/ml_training.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/examples/rl_agent.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/index.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/advanced_patterns.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/ansi_formatting.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/cli-fundamentals.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/cli-patterns.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/configuration-patterns.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/core-concepts.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/environment_variables.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/help-generation.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/hyperparameter_sweeps.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/naming-conventions.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/parameter-iteration.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/parameter-overrides.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/type-system.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/key_concepts/welcome.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/migration.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/quick_start.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/docs/requirements.txt +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/examples/union_subcommands.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/figures/man-page.png +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/figures/params-proto-autocompletion.gif +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/figures/spec_files.png +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/scratch/demo_params_proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/scratch/demo_v3.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/scratch/proto_DAT_scratch.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/scratch/proto_dependency_tree_pattern.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/references/cli-and-types.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/references/environment-vars.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/references/patterns.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/references/sweeps.md +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/app.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/cli/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/cli/ansi_help.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/cli/help_gen.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/documentation.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/envvar.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/hyper/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/hyper/proxies.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/hyper/sweep.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/parse_env_template.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/type_utils.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v1/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v1/hyper.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v1/params_proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v2/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v2/hyper.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v2/partial.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v2/proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/src/params_proto/v2/utils.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v1/__init__.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v1/test_hyper.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v1/test_params_proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_Eval.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_neo_hyper.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_neo_proto.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_neo_proto_cli.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_neo_proto_partial.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v2/test_utils.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/conftest.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/samples/train.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_advanced_types.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_class_level_methods.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_cli_parsing.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_help_strings.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_method_self_param.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_parse_env_template.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_piter.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_positional_example.sh +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_comments.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_core.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_envvar.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_linebreaking.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_partial.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_proto_required.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_strings.py +0 -0
- {params_proto-3.2.0 → params_proto-3.2.1}/tests/test_v3/test_sweep.py +0 -0
|
@@ -4,13 +4,7 @@
|
|
|
4
4
|
<option name="autoReloadType" value="SELECTIVE" />
|
|
5
5
|
</component>
|
|
6
6
|
<component name="ChangeListManager">
|
|
7
|
-
<list default="true" id="7a053ece-f497-4c97-ac58-a86c807155ac" name="Changes" comment="add design specs"
|
|
8
|
-
<change beforePath="$PROJECT_DIR$/src/params_proto/cli/cli_parse.py" beforeDir="false" afterPath="$PROJECT_DIR$/src/params_proto/cli/cli_parse.py" afterDir="false" />
|
|
9
|
-
<change beforePath="$PROJECT_DIR$/src/params_proto/hyper/sweep.py" beforeDir="false" afterPath="$PROJECT_DIR$/src/params_proto/hyper/sweep.py" afterDir="false" />
|
|
10
|
-
<change beforePath="$PROJECT_DIR$/src/params_proto/proto.py" beforeDir="false" afterPath="$PROJECT_DIR$/src/params_proto/proto.py" afterDir="false" />
|
|
11
|
-
<change beforePath="$PROJECT_DIR$/tests/test_v3/test_cli_parsing.py" beforeDir="false" afterPath="$PROJECT_DIR$/tests/test_v3/test_cli_parsing.py" afterDir="false" />
|
|
12
|
-
<change beforePath="$PROJECT_DIR$/tests/test_v3/test_piter.py" beforeDir="false" afterPath="$PROJECT_DIR$/tests/test_v3/test_piter.py" afterDir="false" />
|
|
13
|
-
</list>
|
|
7
|
+
<list default="true" id="7a053ece-f497-4c97-ac58-a86c807155ac" name="Changes" comment="add design specs" />
|
|
14
8
|
<option name="SHOW_DIALOG" value="false" />
|
|
15
9
|
<option name="HIGHLIGHT_CONFLICTS" value="true" />
|
|
16
10
|
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
|
|
@@ -263,7 +257,7 @@
|
|
|
263
257
|
<workItem from="1768382679094" duration="3000" />
|
|
264
258
|
<workItem from="1769135684123" duration="4247000" />
|
|
265
259
|
<workItem from="1769569263076" duration="1863000" />
|
|
266
|
-
<workItem from="1769933004821" duration="
|
|
260
|
+
<workItem from="1769933004821" duration="3502000" />
|
|
267
261
|
</task>
|
|
268
262
|
<task id="LOCAL-00001" summary="add design specs">
|
|
269
263
|
<option name="closed" value="true" />
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: params-proto
|
|
3
|
-
Version: 3.2.
|
|
3
|
+
Version: 3.2.1
|
|
4
4
|
Summary: Modern Hyper Parameter Management for Machine Learning
|
|
5
5
|
Project-URL: Homepage, https://github.com/geyang/params-proto
|
|
6
6
|
Project-URL: Documentation, https://params-proto.readthedocs.io
|
|
@@ -61,10 +61,11 @@ def main(model: Model):
|
|
|
61
61
|
```
|
|
62
62
|
|
|
63
63
|
```python
|
|
64
|
-
# Pattern 3: Optional simple parameters
|
|
65
|
-
|
|
64
|
+
# Pattern 3: Optional simple parameters
|
|
65
|
+
from typing import Optional
|
|
66
|
+
|
|
66
67
|
@proto.cli
|
|
67
|
-
def process(checkpoint: str = None, batch_size: int = 32):
|
|
68
|
+
def process(checkpoint: Optional[str] = None, batch_size: int = 32):
|
|
68
69
|
pass
|
|
69
70
|
|
|
70
71
|
# CLI: python process.py --checkpoint model.pt
|
|
@@ -134,17 +135,18 @@ if __name__ == "__main__":
|
|
|
134
135
|
`Optional[T]` is for parameters that **may or may not be provided**:
|
|
135
136
|
|
|
136
137
|
```python
|
|
138
|
+
from typing import Optional
|
|
139
|
+
|
|
137
140
|
@proto.cli
|
|
138
141
|
def train(
|
|
139
|
-
checkpoint: str = None,
|
|
140
|
-
# checkpoint: Optional[str] = None, # ⚠️ Doesn't fully work yet
|
|
142
|
+
checkpoint: Optional[str] = None, # Optional with None default
|
|
141
143
|
epochs: int = 10,
|
|
142
144
|
):
|
|
143
145
|
"""Train model."""
|
|
144
146
|
pass
|
|
145
147
|
```
|
|
146
148
|
|
|
147
|
-
**
|
|
149
|
+
**CLI usage:**
|
|
148
150
|
|
|
149
151
|
```bash
|
|
150
152
|
python train.py --checkpoint model.pt # Provide value
|
|
@@ -152,21 +154,15 @@ python train.py # Omit for None default
|
|
|
152
154
|
python train.py --checkpoint model.pt --epochs 50
|
|
153
155
|
```
|
|
154
156
|
|
|
155
|
-
|
|
156
|
-
**Current limitation:** `Optional[str]`, `Optional[int]`, etc. are not fully supported.
|
|
157
|
-
Use regular parameters with defaults as a workaround:
|
|
157
|
+
Both `Optional[str]` and `str = None` work equivalently:
|
|
158
158
|
|
|
159
159
|
```python
|
|
160
|
-
#
|
|
160
|
+
# Both work
|
|
161
161
|
@proto.cli
|
|
162
|
-
def train(checkpoint: str = None
|
|
163
|
-
pass
|
|
162
|
+
def train(checkpoint: Optional[str] = None): ...
|
|
164
163
|
|
|
165
|
-
# ⚠️ Doesn't work yet
|
|
166
164
|
@proto.cli
|
|
167
|
-
def train(checkpoint:
|
|
168
|
-
pass
|
|
169
|
-
```
|
|
165
|
+
def train(checkpoint: str = None): ...
|
|
170
166
|
```
|
|
171
167
|
|
|
172
168
|
## Key Differences
|
|
@@ -174,8 +170,8 @@ def train(checkpoint: Optional[str] = None, epochs: int = 10):
|
|
|
174
170
|
| Type | Purpose | CLI Syntax | When to Use |
|
|
175
171
|
|------|---------|-----------|-------------|
|
|
176
172
|
| `Union[ClassA, ClassB]` | Choose which class instance | `--param:ClassName` or positional | Multiple configurations (optimizers, models, etc.) |
|
|
177
|
-
| `Optional[str]` | Value may or may not be provided | `--param value` | Optional simple parameters
|
|
178
|
-
| `str
|
|
173
|
+
| `Optional[str]` | Value may or may not be provided | `--param value` | Optional simple parameters |
|
|
174
|
+
| `str = None` | Same as Optional | `--param value` | Alternative syntax for optional parameters |
|
|
179
175
|
|
|
180
176
|
## Syntax Variations
|
|
181
177
|
|
|
@@ -2,6 +2,22 @@
|
|
|
2
2
|
|
|
3
3
|
This page contains the release history and changelog for params-proto.
|
|
4
4
|
|
|
5
|
+
## Version 3.2.1 (2025-02-01)
|
|
6
|
+
|
|
7
|
+
### 🐛 Bug Fixes
|
|
8
|
+
|
|
9
|
+
- **Positional Arguments in Subcommands**: Fixed positional arguments being silently ignored after subcommand name
|
|
10
|
+
- Before: `myapp add my-env/v1.2.3` → positional arg silently dropped
|
|
11
|
+
- After: `myapp add my-env/v1.2.3` → positional arg captured by subcommand's required field
|
|
12
|
+
- Enables CLI patterns like `pip install requests`, `cargo add serde`
|
|
13
|
+
- Raises clear error for extra unrecognized positional arguments
|
|
14
|
+
|
|
15
|
+
### 📚 Documentation
|
|
16
|
+
|
|
17
|
+
- Fixed outdated notes claiming `Optional[str]` was not supported (it works)
|
|
18
|
+
|
|
19
|
+
---
|
|
20
|
+
|
|
5
21
|
## Version 3.2.0 (2025-02-01)
|
|
6
22
|
|
|
7
23
|
### ✨ Features
|
|
Binary file
|
|
@@ -57,6 +57,32 @@ def _normalize_class_name(class_name: str) -> str:
|
|
|
57
57
|
return class_name.replace("-", "").replace("_", "").lower()
|
|
58
58
|
|
|
59
59
|
|
|
60
|
+
def _get_required_fields(cls) -> List[str]:
|
|
61
|
+
"""Get list of required field names (fields without defaults) in order."""
|
|
62
|
+
import dataclasses
|
|
63
|
+
|
|
64
|
+
required = []
|
|
65
|
+
|
|
66
|
+
# Check if it's a dataclass
|
|
67
|
+
if dataclasses.is_dataclass(cls):
|
|
68
|
+
for field in dataclasses.fields(cls):
|
|
69
|
+
has_default = (
|
|
70
|
+
field.default is not dataclasses.MISSING
|
|
71
|
+
or field.default_factory is not dataclasses.MISSING
|
|
72
|
+
)
|
|
73
|
+
if not has_default:
|
|
74
|
+
required.append(field.name)
|
|
75
|
+
else:
|
|
76
|
+
# For regular classes, check annotations and class-level defaults
|
|
77
|
+
annotations = getattr(cls, "__annotations__", {})
|
|
78
|
+
for name in annotations:
|
|
79
|
+
if not hasattr(cls, name):
|
|
80
|
+
# No class-level default
|
|
81
|
+
required.append(name)
|
|
82
|
+
|
|
83
|
+
return required
|
|
84
|
+
|
|
85
|
+
|
|
60
86
|
def _match_class_by_name(name: str, classes: list) -> Union[type, None]:
|
|
61
87
|
"""Match a string to one of the Union classes.
|
|
62
88
|
|
|
@@ -187,6 +213,8 @@ def parse_cli_args(wrapper) -> Dict[str, Any]:
|
|
|
187
213
|
positional_values = []
|
|
188
214
|
union_selections = {} # param_name -> selected_class
|
|
189
215
|
union_attrs = {} # (param_name, attr_name) -> value
|
|
216
|
+
union_positional = {} # param_name -> [positional_args] for subcommand fields
|
|
217
|
+
current_union_param = None # Track which union we're collecting positionals for
|
|
190
218
|
|
|
191
219
|
args = sys.argv[1:]
|
|
192
220
|
i = 0
|
|
@@ -377,12 +405,18 @@ def parse_cli_args(wrapper) -> Dict[str, Any]:
|
|
|
377
405
|
selected_class = _match_class_by_name(arg, union_classes)
|
|
378
406
|
if selected_class:
|
|
379
407
|
union_selections[param_name] = selected_class
|
|
408
|
+
current_union_param = param_name # Track for following positionals
|
|
409
|
+
union_positional[param_name] = []
|
|
380
410
|
matched_union = True
|
|
381
411
|
i += 1
|
|
382
412
|
break
|
|
383
413
|
|
|
384
414
|
if not matched_union:
|
|
385
|
-
|
|
415
|
+
# If we have a current union, add positional to its list
|
|
416
|
+
if current_union_param is not None:
|
|
417
|
+
union_positional[current_union_param].append(arg)
|
|
418
|
+
else:
|
|
419
|
+
positional_values.append(arg)
|
|
386
420
|
i += 1
|
|
387
421
|
|
|
388
422
|
# Assign positional arguments to required parameters
|
|
@@ -433,6 +467,33 @@ def parse_cli_args(wrapper) -> Dict[str, Any]:
|
|
|
433
467
|
# No annotations, treat as string
|
|
434
468
|
attrs[attr_name] = value_str
|
|
435
469
|
|
|
470
|
+
# Assign positional args to required fields of the selected class
|
|
471
|
+
if param_name in union_positional and union_positional[param_name]:
|
|
472
|
+
positionals = union_positional[param_name]
|
|
473
|
+
required_fields = _get_required_fields(selected_class)
|
|
474
|
+
|
|
475
|
+
for field_idx, field_name in enumerate(required_fields):
|
|
476
|
+
if field_name in attrs:
|
|
477
|
+
# Already set by named arg, skip
|
|
478
|
+
continue
|
|
479
|
+
if field_idx < len(positionals):
|
|
480
|
+
# Get type annotation for conversion
|
|
481
|
+
if hasattr(selected_class, "__annotations__"):
|
|
482
|
+
field_type = selected_class.__annotations__.get(field_name, str)
|
|
483
|
+
try:
|
|
484
|
+
attrs[field_name] = _convert_type(positionals[field_idx], field_type)
|
|
485
|
+
except (ValueError, TypeError):
|
|
486
|
+
raise SystemExit(
|
|
487
|
+
f"error: invalid value for {field_name}: {positionals[field_idx]}"
|
|
488
|
+
)
|
|
489
|
+
else:
|
|
490
|
+
attrs[field_name] = positionals[field_idx]
|
|
491
|
+
|
|
492
|
+
# Check for extra positional args
|
|
493
|
+
if len(positionals) > len(required_fields):
|
|
494
|
+
extra = positionals[len(required_fields):]
|
|
495
|
+
raise SystemExit(f"error: unrecognized arguments: {' '.join(extra)}")
|
|
496
|
+
|
|
436
497
|
# If selected_class is a proto.prefix singleton, merge its overrides
|
|
437
498
|
from params_proto.proto import _SINGLETONS, ptype
|
|
438
499
|
|
|
@@ -447,6 +508,14 @@ def parse_cli_args(wrapper) -> Dict[str, Any]:
|
|
|
447
508
|
attrs[key] = value
|
|
448
509
|
break
|
|
449
510
|
|
|
511
|
+
# Check for missing required fields
|
|
512
|
+
required_fields = _get_required_fields(selected_class)
|
|
513
|
+
for field_name in required_fields:
|
|
514
|
+
if field_name not in attrs:
|
|
515
|
+
raise SystemExit(
|
|
516
|
+
f"error: {selected_class.__name__} requires argument: {field_name}"
|
|
517
|
+
)
|
|
518
|
+
|
|
450
519
|
# Instantiate the class with collected attributes
|
|
451
520
|
try:
|
|
452
521
|
instance = selected_class(**attrs)
|
|
@@ -816,3 +816,141 @@ class TestProtoPrefixRequiresPrefix:
|
|
|
816
816
|
# Unprefixed should error for @proto.prefix class
|
|
817
817
|
result = run_cli(script, ["train-config", "--epochs", "200"], expect_error=True)
|
|
818
818
|
assert "unrecognized argument" in result["stderr"]
|
|
819
|
+
|
|
820
|
+
|
|
821
|
+
class TestPositionalArgsInSubcommands:
|
|
822
|
+
"""Test positional arguments captured by subcommand fields.
|
|
823
|
+
|
|
824
|
+
This tests the pattern: myapp add my-env/v1.2.3
|
|
825
|
+
Where 'add' is the subcommand and 'my-env/v1.2.3' is captured by the
|
|
826
|
+
subcommand's first required field.
|
|
827
|
+
"""
|
|
828
|
+
|
|
829
|
+
def test_positional_arg_captured_by_subcommand_field(self, run_cli):
|
|
830
|
+
"""Test positional arg after subcommand name is captured."""
|
|
831
|
+
script = dedent("""
|
|
832
|
+
from dataclasses import dataclass
|
|
833
|
+
from params_proto import proto
|
|
834
|
+
|
|
835
|
+
@dataclass
|
|
836
|
+
class AddCommand:
|
|
837
|
+
env: str # Required field, no default
|
|
838
|
+
|
|
839
|
+
@dataclass
|
|
840
|
+
class RemoveCommand:
|
|
841
|
+
env: str # Required field
|
|
842
|
+
|
|
843
|
+
@proto.cli
|
|
844
|
+
def main(command: AddCommand | RemoveCommand):
|
|
845
|
+
print(f"command={command.__class__.__name__}")
|
|
846
|
+
print(f"env={command.env}")
|
|
847
|
+
|
|
848
|
+
if __name__ == "__main__":
|
|
849
|
+
main()
|
|
850
|
+
""")
|
|
851
|
+
|
|
852
|
+
# Positional arg after subcommand should be captured by 'env' field
|
|
853
|
+
result = run_cli(script, ["add", "my-env/v1.2.3"])
|
|
854
|
+
lines = result["stdout"].strip().split("\n")
|
|
855
|
+
assert lines[0] == "command=AddCommand"
|
|
856
|
+
assert lines[1] == "env=my-env/v1.2.3"
|
|
857
|
+
|
|
858
|
+
result = run_cli(script, ["remove", "old-env/v0.9.0"])
|
|
859
|
+
lines = result["stdout"].strip().split("\n")
|
|
860
|
+
assert lines[0] == "command=RemoveCommand"
|
|
861
|
+
assert lines[1] == "env=old-env/v0.9.0"
|
|
862
|
+
|
|
863
|
+
def test_multiple_positional_args_in_subcommand(self, run_cli):
|
|
864
|
+
"""Test multiple positional args captured by subcommand fields."""
|
|
865
|
+
script = dedent("""
|
|
866
|
+
from dataclasses import dataclass
|
|
867
|
+
from params_proto import proto
|
|
868
|
+
|
|
869
|
+
@dataclass
|
|
870
|
+
class InstallCommand:
|
|
871
|
+
package: str # Required, first positional
|
|
872
|
+
version: str # Required, second positional
|
|
873
|
+
|
|
874
|
+
@proto.cli
|
|
875
|
+
def main(command: InstallCommand):
|
|
876
|
+
print(f"package={command.package}")
|
|
877
|
+
print(f"version={command.version}")
|
|
878
|
+
|
|
879
|
+
if __name__ == "__main__":
|
|
880
|
+
main()
|
|
881
|
+
""")
|
|
882
|
+
|
|
883
|
+
result = run_cli(script, ["install", "requests", "2.28.0"])
|
|
884
|
+
lines = result["stdout"].strip().split("\n")
|
|
885
|
+
assert lines[0] == "package=requests"
|
|
886
|
+
assert lines[1] == "version=2.28.0"
|
|
887
|
+
|
|
888
|
+
def test_positional_with_named_args(self, run_cli):
|
|
889
|
+
"""Test mixing positional and named args in subcommand."""
|
|
890
|
+
script = dedent("""
|
|
891
|
+
from dataclasses import dataclass
|
|
892
|
+
from params_proto import proto
|
|
893
|
+
|
|
894
|
+
@dataclass
|
|
895
|
+
class CloneCommand:
|
|
896
|
+
url: str # Required positional
|
|
897
|
+
depth: int = 0 # Optional with default
|
|
898
|
+
|
|
899
|
+
@proto.cli
|
|
900
|
+
def main(command: CloneCommand):
|
|
901
|
+
print(f"url={command.url}")
|
|
902
|
+
print(f"depth={command.depth}")
|
|
903
|
+
|
|
904
|
+
if __name__ == "__main__":
|
|
905
|
+
main()
|
|
906
|
+
""")
|
|
907
|
+
|
|
908
|
+
# Positional url, named depth
|
|
909
|
+
result = run_cli(script, ["clone", "https://github.com/test", "--depth", "1"])
|
|
910
|
+
lines = result["stdout"].strip().split("\n")
|
|
911
|
+
assert lines[0] == "url=https://github.com/test"
|
|
912
|
+
assert lines[1] == "depth=1"
|
|
913
|
+
|
|
914
|
+
def test_positional_arg_missing_should_error(self, run_cli):
|
|
915
|
+
"""Test that missing required positional arg errors."""
|
|
916
|
+
script = dedent("""
|
|
917
|
+
from dataclasses import dataclass
|
|
918
|
+
from params_proto import proto
|
|
919
|
+
|
|
920
|
+
@dataclass
|
|
921
|
+
class AddCommand:
|
|
922
|
+
env: str # Required field
|
|
923
|
+
|
|
924
|
+
@proto.cli
|
|
925
|
+
def main(command: AddCommand):
|
|
926
|
+
print(f"env={command.env}")
|
|
927
|
+
|
|
928
|
+
if __name__ == "__main__":
|
|
929
|
+
main()
|
|
930
|
+
""")
|
|
931
|
+
|
|
932
|
+
# Missing required positional should error
|
|
933
|
+
result = run_cli(script, ["add"], expect_error=True)
|
|
934
|
+
assert result["returncode"] != 0
|
|
935
|
+
|
|
936
|
+
def test_extra_positional_should_error(self, run_cli):
|
|
937
|
+
"""Test that extra unrecognized positional args error."""
|
|
938
|
+
script = dedent("""
|
|
939
|
+
from dataclasses import dataclass
|
|
940
|
+
from params_proto import proto
|
|
941
|
+
|
|
942
|
+
@dataclass
|
|
943
|
+
class AddCommand:
|
|
944
|
+
env: str # Required field
|
|
945
|
+
|
|
946
|
+
@proto.cli
|
|
947
|
+
def main(command: AddCommand):
|
|
948
|
+
print(f"env={command.env}")
|
|
949
|
+
|
|
950
|
+
if __name__ == "__main__":
|
|
951
|
+
main()
|
|
952
|
+
""")
|
|
953
|
+
|
|
954
|
+
# Extra positional arg should error
|
|
955
|
+
result = run_cli(script, ["add", "env1", "extra-arg"], expect_error=True)
|
|
956
|
+
assert result["returncode"] != 0
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/environment_variables.md
RENAMED
|
File without changes
|
{params_proto-3.2.0 → params_proto-3.2.1}/docs/_archive_v2/examples/hyperparameter_sweeps.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{params_proto-3.2.0 → params_proto-3.2.1}/skills/params-proto/references/environment-vars.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|