webscout 8.3.7__py3-none-any.whl → 2025.10.11__py3-none-any.whl
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.
Potentially problematic release.
This version of webscout might be problematic. Click here for more details.
- webscout/AIauto.py +250 -250
- webscout/AIbase.py +379 -379
- webscout/AIutel.py +60 -60
- webscout/Bard.py +1012 -1012
- webscout/Bing_search.py +417 -417
- webscout/DWEBS.py +529 -529
- webscout/Extra/Act.md +309 -309
- webscout/Extra/GitToolkit/__init__.py +10 -10
- webscout/Extra/GitToolkit/gitapi/README.md +110 -110
- webscout/Extra/GitToolkit/gitapi/__init__.py +11 -11
- webscout/Extra/GitToolkit/gitapi/repository.py +195 -195
- webscout/Extra/GitToolkit/gitapi/user.py +96 -96
- webscout/Extra/GitToolkit/gitapi/utils.py +61 -61
- webscout/Extra/YTToolkit/README.md +375 -375
- webscout/Extra/YTToolkit/YTdownloader.py +956 -956
- webscout/Extra/YTToolkit/__init__.py +2 -2
- webscout/Extra/YTToolkit/transcriber.py +475 -475
- webscout/Extra/YTToolkit/ytapi/README.md +44 -44
- webscout/Extra/YTToolkit/ytapi/__init__.py +6 -6
- webscout/Extra/YTToolkit/ytapi/channel.py +307 -307
- webscout/Extra/YTToolkit/ytapi/errors.py +13 -13
- webscout/Extra/YTToolkit/ytapi/extras.py +118 -118
- webscout/Extra/YTToolkit/ytapi/https.py +88 -88
- webscout/Extra/YTToolkit/ytapi/patterns.py +61 -61
- webscout/Extra/YTToolkit/ytapi/playlist.py +58 -58
- webscout/Extra/YTToolkit/ytapi/pool.py +7 -7
- webscout/Extra/YTToolkit/ytapi/query.py +39 -39
- webscout/Extra/YTToolkit/ytapi/stream.py +62 -62
- webscout/Extra/YTToolkit/ytapi/utils.py +62 -62
- webscout/Extra/YTToolkit/ytapi/video.py +232 -232
- webscout/Extra/autocoder/__init__.py +9 -9
- webscout/Extra/autocoder/autocoder.py +1105 -1105
- webscout/Extra/autocoder/autocoder_utiles.py +332 -332
- webscout/Extra/gguf.md +429 -429
- webscout/Extra/gguf.py +1213 -1213
- webscout/Extra/tempmail/README.md +487 -487
- webscout/Extra/tempmail/__init__.py +27 -27
- webscout/Extra/tempmail/async_utils.py +140 -140
- webscout/Extra/tempmail/base.py +160 -160
- webscout/Extra/tempmail/cli.py +186 -186
- webscout/Extra/tempmail/emailnator.py +84 -84
- webscout/Extra/tempmail/mail_tm.py +360 -360
- webscout/Extra/tempmail/temp_mail_io.py +291 -291
- webscout/Extra/weather.md +281 -281
- webscout/Extra/weather.py +193 -193
- webscout/Litlogger/README.md +10 -10
- webscout/Litlogger/__init__.py +15 -15
- webscout/Litlogger/formats.py +13 -13
- webscout/Litlogger/handlers.py +121 -121
- webscout/Litlogger/levels.py +13 -13
- webscout/Litlogger/logger.py +134 -134
- webscout/Provider/AISEARCH/Perplexity.py +332 -332
- webscout/Provider/AISEARCH/README.md +279 -279
- webscout/Provider/AISEARCH/__init__.py +16 -1
- webscout/Provider/AISEARCH/felo_search.py +206 -206
- webscout/Provider/AISEARCH/genspark_search.py +323 -323
- webscout/Provider/AISEARCH/hika_search.py +185 -185
- webscout/Provider/AISEARCH/iask_search.py +410 -410
- webscout/Provider/AISEARCH/monica_search.py +219 -219
- webscout/Provider/AISEARCH/scira_search.py +316 -316
- webscout/Provider/AISEARCH/stellar_search.py +177 -177
- webscout/Provider/AISEARCH/webpilotai_search.py +255 -255
- webscout/Provider/Aitopia.py +314 -314
- webscout/Provider/Apriel.py +306 -0
- webscout/Provider/ChatGPTClone.py +236 -236
- webscout/Provider/ChatSandbox.py +343 -343
- webscout/Provider/Cloudflare.py +324 -324
- webscout/Provider/Cohere.py +208 -208
- webscout/Provider/Deepinfra.py +370 -366
- webscout/Provider/ExaAI.py +260 -260
- webscout/Provider/ExaChat.py +308 -308
- webscout/Provider/Flowith.py +221 -221
- webscout/Provider/GMI.py +293 -0
- webscout/Provider/Gemini.py +164 -164
- webscout/Provider/GeminiProxy.py +167 -167
- webscout/Provider/GithubChat.py +371 -372
- webscout/Provider/Groq.py +800 -800
- webscout/Provider/HeckAI.py +383 -383
- webscout/Provider/Jadve.py +282 -282
- webscout/Provider/K2Think.py +307 -307
- webscout/Provider/Koboldai.py +205 -205
- webscout/Provider/LambdaChat.py +423 -423
- webscout/Provider/Nemotron.py +244 -244
- webscout/Provider/Netwrck.py +248 -248
- webscout/Provider/OLLAMA.py +395 -395
- webscout/Provider/OPENAI/Cloudflare.py +393 -393
- webscout/Provider/OPENAI/FalconH1.py +451 -451
- webscout/Provider/OPENAI/FreeGemini.py +296 -296
- webscout/Provider/OPENAI/K2Think.py +431 -431
- webscout/Provider/OPENAI/NEMOTRON.py +240 -240
- webscout/Provider/OPENAI/PI.py +427 -427
- webscout/Provider/OPENAI/README.md +959 -959
- webscout/Provider/OPENAI/TogetherAI.py +345 -345
- webscout/Provider/OPENAI/TwoAI.py +465 -465
- webscout/Provider/OPENAI/__init__.py +33 -18
- webscout/Provider/OPENAI/base.py +248 -248
- webscout/Provider/OPENAI/chatglm.py +528 -0
- webscout/Provider/OPENAI/chatgpt.py +592 -592
- webscout/Provider/OPENAI/chatgptclone.py +521 -521
- webscout/Provider/OPENAI/chatsandbox.py +202 -202
- webscout/Provider/OPENAI/deepinfra.py +318 -314
- webscout/Provider/OPENAI/e2b.py +1665 -1665
- webscout/Provider/OPENAI/exaai.py +420 -420
- webscout/Provider/OPENAI/exachat.py +452 -452
- webscout/Provider/OPENAI/friendli.py +232 -232
- webscout/Provider/OPENAI/{refact.py → gmi.py} +324 -274
- webscout/Provider/OPENAI/groq.py +364 -364
- webscout/Provider/OPENAI/heckai.py +314 -314
- webscout/Provider/OPENAI/llmchatco.py +337 -337
- webscout/Provider/OPENAI/netwrck.py +355 -355
- webscout/Provider/OPENAI/oivscode.py +290 -290
- webscout/Provider/OPENAI/opkfc.py +518 -518
- webscout/Provider/OPENAI/pydantic_imports.py +1 -1
- webscout/Provider/OPENAI/scirachat.py +535 -535
- webscout/Provider/OPENAI/sonus.py +308 -308
- webscout/Provider/OPENAI/standardinput.py +442 -442
- webscout/Provider/OPENAI/textpollinations.py +340 -340
- webscout/Provider/OPENAI/toolbaz.py +419 -416
- webscout/Provider/OPENAI/typefully.py +362 -362
- webscout/Provider/OPENAI/utils.py +295 -295
- webscout/Provider/OPENAI/venice.py +436 -436
- webscout/Provider/OPENAI/wisecat.py +387 -387
- webscout/Provider/OPENAI/writecream.py +166 -166
- webscout/Provider/OPENAI/x0gpt.py +378 -378
- webscout/Provider/OPENAI/yep.py +389 -389
- webscout/Provider/OpenGPT.py +230 -230
- webscout/Provider/Openai.py +243 -243
- webscout/Provider/PI.py +405 -405
- webscout/Provider/Perplexitylabs.py +430 -430
- webscout/Provider/QwenLM.py +272 -272
- webscout/Provider/STT/__init__.py +16 -1
- webscout/Provider/Sambanova.py +257 -257
- webscout/Provider/StandardInput.py +309 -309
- webscout/Provider/TTI/README.md +82 -82
- webscout/Provider/TTI/__init__.py +33 -18
- webscout/Provider/TTI/aiarta.py +413 -413
- webscout/Provider/TTI/base.py +136 -136
- webscout/Provider/TTI/bing.py +243 -243
- webscout/Provider/TTI/gpt1image.py +149 -149
- webscout/Provider/TTI/imagen.py +196 -196
- webscout/Provider/TTI/infip.py +211 -211
- webscout/Provider/TTI/magicstudio.py +232 -232
- webscout/Provider/TTI/monochat.py +219 -219
- webscout/Provider/TTI/piclumen.py +214 -214
- webscout/Provider/TTI/pixelmuse.py +232 -232
- webscout/Provider/TTI/pollinations.py +232 -232
- webscout/Provider/TTI/together.py +288 -288
- webscout/Provider/TTI/utils.py +12 -12
- webscout/Provider/TTI/venice.py +367 -367
- webscout/Provider/TTS/README.md +192 -192
- webscout/Provider/TTS/__init__.py +33 -18
- webscout/Provider/TTS/parler.py +110 -110
- webscout/Provider/TTS/streamElements.py +333 -333
- webscout/Provider/TTS/utils.py +280 -280
- webscout/Provider/TeachAnything.py +237 -237
- webscout/Provider/TextPollinationsAI.py +310 -310
- webscout/Provider/TogetherAI.py +356 -356
- webscout/Provider/TwoAI.py +312 -312
- webscout/Provider/TypliAI.py +311 -311
- webscout/Provider/UNFINISHED/ChatHub.py +208 -208
- webscout/Provider/UNFINISHED/ChutesAI.py +313 -313
- webscout/Provider/UNFINISHED/GizAI.py +294 -294
- webscout/Provider/UNFINISHED/Marcus.py +198 -198
- webscout/Provider/UNFINISHED/Qodo.py +477 -477
- webscout/Provider/UNFINISHED/VercelAIGateway.py +338 -338
- webscout/Provider/UNFINISHED/XenAI.py +324 -324
- webscout/Provider/UNFINISHED/Youchat.py +330 -330
- webscout/Provider/UNFINISHED/liner.py +334 -0
- webscout/Provider/UNFINISHED/liner_api_request.py +262 -262
- webscout/Provider/UNFINISHED/puterjs.py +634 -634
- webscout/Provider/UNFINISHED/samurai.py +223 -223
- webscout/Provider/UNFINISHED/test_lmarena.py +119 -119
- webscout/Provider/Venice.py +250 -250
- webscout/Provider/VercelAI.py +256 -256
- webscout/Provider/WiseCat.py +231 -231
- webscout/Provider/WrDoChat.py +366 -366
- webscout/Provider/__init__.py +33 -18
- webscout/Provider/ai4chat.py +174 -174
- webscout/Provider/akashgpt.py +331 -331
- webscout/Provider/cerebras.py +446 -446
- webscout/Provider/chatglm.py +394 -301
- webscout/Provider/cleeai.py +211 -211
- webscout/Provider/elmo.py +282 -282
- webscout/Provider/geminiapi.py +208 -208
- webscout/Provider/granite.py +261 -261
- webscout/Provider/hermes.py +263 -263
- webscout/Provider/julius.py +223 -223
- webscout/Provider/learnfastai.py +309 -309
- webscout/Provider/llama3mitril.py +214 -214
- webscout/Provider/llmchat.py +243 -243
- webscout/Provider/llmchatco.py +290 -290
- webscout/Provider/meta.py +801 -801
- webscout/Provider/oivscode.py +309 -309
- webscout/Provider/scira_chat.py +383 -383
- webscout/Provider/searchchat.py +292 -292
- webscout/Provider/sonus.py +258 -258
- webscout/Provider/toolbaz.py +370 -367
- webscout/Provider/turboseek.py +273 -273
- webscout/Provider/typefully.py +207 -207
- webscout/Provider/yep.py +372 -372
- webscout/__init__.py +30 -31
- webscout/__main__.py +5 -5
- webscout/auth/api_key_manager.py +189 -189
- webscout/auth/config.py +175 -175
- webscout/auth/models.py +185 -185
- webscout/auth/routes.py +664 -664
- webscout/auth/simple_logger.py +236 -236
- webscout/cli.py +523 -523
- webscout/conversation.py +438 -438
- webscout/exceptions.py +361 -361
- webscout/litagent/Readme.md +298 -298
- webscout/litagent/__init__.py +28 -28
- webscout/litagent/agent.py +581 -581
- webscout/litagent/constants.py +59 -59
- webscout/litprinter/__init__.py +58 -58
- webscout/models.py +181 -181
- webscout/optimizers.py +419 -419
- webscout/prompt_manager.py +288 -288
- webscout/sanitize.py +1078 -1078
- webscout/scout/README.md +401 -401
- webscout/scout/__init__.py +8 -8
- webscout/scout/core/__init__.py +6 -6
- webscout/scout/core/crawler.py +297 -297
- webscout/scout/core/scout.py +706 -706
- webscout/scout/core/search_result.py +95 -95
- webscout/scout/core/text_analyzer.py +62 -62
- webscout/scout/core/text_utils.py +277 -277
- webscout/scout/core/web_analyzer.py +51 -51
- webscout/scout/element.py +599 -599
- webscout/scout/parsers/__init__.py +69 -69
- webscout/scout/parsers/html5lib_parser.py +172 -172
- webscout/scout/parsers/html_parser.py +236 -236
- webscout/scout/parsers/lxml_parser.py +178 -178
- webscout/scout/utils.py +37 -37
- webscout/swiftcli/Readme.md +323 -323
- webscout/swiftcli/__init__.py +95 -95
- webscout/swiftcli/core/__init__.py +7 -7
- webscout/swiftcli/core/cli.py +308 -308
- webscout/swiftcli/core/context.py +104 -104
- webscout/swiftcli/core/group.py +241 -241
- webscout/swiftcli/decorators/__init__.py +28 -28
- webscout/swiftcli/decorators/command.py +221 -221
- webscout/swiftcli/decorators/options.py +220 -220
- webscout/swiftcli/decorators/output.py +302 -302
- webscout/swiftcli/exceptions.py +21 -21
- webscout/swiftcli/plugins/__init__.py +9 -9
- webscout/swiftcli/plugins/base.py +135 -135
- webscout/swiftcli/plugins/manager.py +269 -269
- webscout/swiftcli/utils/__init__.py +59 -59
- webscout/swiftcli/utils/formatting.py +252 -252
- webscout/swiftcli/utils/parsing.py +267 -267
- webscout/update_checker.py +117 -117
- webscout/version.py +1 -1
- webscout/webscout_search.py +1183 -1183
- webscout/webscout_search_async.py +649 -649
- webscout/yep_search.py +346 -346
- webscout/zeroart/README.md +89 -89
- webscout/zeroart/__init__.py +134 -134
- webscout/zeroart/base.py +66 -66
- webscout/zeroart/effects.py +100 -100
- webscout/zeroart/fonts.py +1238 -1238
- {webscout-8.3.7.dist-info → webscout-2025.10.11.dist-info}/METADATA +937 -937
- webscout-2025.10.11.dist-info/RECORD +300 -0
- webscout/Provider/AISEARCH/DeepFind.py +0 -254
- webscout/Provider/OPENAI/Qwen3.py +0 -303
- webscout/Provider/OPENAI/qodo.py +0 -630
- webscout/Provider/OPENAI/xenai.py +0 -514
- webscout/tempid.py +0 -134
- webscout-8.3.7.dist-info/RECORD +0 -301
- {webscout-8.3.7.dist-info → webscout-2025.10.11.dist-info}/WHEEL +0 -0
- {webscout-8.3.7.dist-info → webscout-2025.10.11.dist-info}/entry_points.txt +0 -0
- {webscout-8.3.7.dist-info → webscout-2025.10.11.dist-info}/licenses/LICENSE.md +0 -0
- {webscout-8.3.7.dist-info → webscout-2025.10.11.dist-info}/top_level.txt +0 -0
|
@@ -1,267 +1,267 @@
|
|
|
1
|
-
"""Utility functions for parsing and validating command-line arguments."""
|
|
2
|
-
|
|
3
|
-
import os
|
|
4
|
-
import json
|
|
5
|
-
import yaml
|
|
6
|
-
from pathlib import Path
|
|
7
|
-
from typing import Any, Dict, List, Optional, Union, Type
|
|
8
|
-
|
|
9
|
-
from ..exceptions import BadParameter, UsageError
|
|
10
|
-
|
|
11
|
-
def parse_args(args: List[str]) -> Dict[str, Any]:
|
|
12
|
-
"""
|
|
13
|
-
Parse command line arguments into a dictionary.
|
|
14
|
-
|
|
15
|
-
Args:
|
|
16
|
-
args: List of command line arguments
|
|
17
|
-
|
|
18
|
-
Returns:
|
|
19
|
-
Dictionary of parsed arguments
|
|
20
|
-
|
|
21
|
-
Example:
|
|
22
|
-
>>> parse_args(['--name', 'test', '--flag', '-n', '42'])
|
|
23
|
-
{'name': 'test', 'flag': True, 'n': '42'}
|
|
24
|
-
"""
|
|
25
|
-
parsed = {}
|
|
26
|
-
i = 0
|
|
27
|
-
while i < len(args):
|
|
28
|
-
arg = args[i]
|
|
29
|
-
|
|
30
|
-
# Handle flags/options
|
|
31
|
-
if arg.startswith('-'):
|
|
32
|
-
key = arg.lstrip('-').replace('-', '_')
|
|
33
|
-
|
|
34
|
-
# Check if next arg is a value or another flag
|
|
35
|
-
if i + 1 >= len(args) or args[i + 1].startswith('-'):
|
|
36
|
-
parsed[key] = True # Flag without value
|
|
37
|
-
else:
|
|
38
|
-
parsed[key] = args[i + 1]
|
|
39
|
-
i += 1
|
|
40
|
-
else:
|
|
41
|
-
# Positional argument
|
|
42
|
-
parsed[f'arg{len([k for k in parsed.keys() if k.startswith("arg")])}'] = arg
|
|
43
|
-
|
|
44
|
-
i += 1
|
|
45
|
-
|
|
46
|
-
return parsed
|
|
47
|
-
|
|
48
|
-
def validate_required(
|
|
49
|
-
params: Dict[str, Any],
|
|
50
|
-
required: List[str]
|
|
51
|
-
) -> None:
|
|
52
|
-
"""
|
|
53
|
-
Validate required parameters are present.
|
|
54
|
-
|
|
55
|
-
Args:
|
|
56
|
-
params: Parameter dictionary
|
|
57
|
-
required: List of required parameter names
|
|
58
|
-
|
|
59
|
-
Raises:
|
|
60
|
-
UsageError: If required parameter is missing
|
|
61
|
-
"""
|
|
62
|
-
missing = [p for p in required if p not in params]
|
|
63
|
-
if missing:
|
|
64
|
-
raise UsageError(f"Missing required parameters: {', '.join(missing)}")
|
|
65
|
-
|
|
66
|
-
def convert_type(
|
|
67
|
-
value: str,
|
|
68
|
-
type_: Type,
|
|
69
|
-
param_name: str
|
|
70
|
-
) -> Any:
|
|
71
|
-
"""
|
|
72
|
-
Convert string value to specified type.
|
|
73
|
-
|
|
74
|
-
Args:
|
|
75
|
-
value: String value to convert
|
|
76
|
-
type_: Target type
|
|
77
|
-
param_name: Parameter name for error messages
|
|
78
|
-
|
|
79
|
-
Returns:
|
|
80
|
-
Converted value
|
|
81
|
-
|
|
82
|
-
Raises:
|
|
83
|
-
BadParameter: If conversion fails
|
|
84
|
-
"""
|
|
85
|
-
try:
|
|
86
|
-
if type_ == bool:
|
|
87
|
-
return value.lower() in ('true', 't', 'yes', 'y', '1')
|
|
88
|
-
return type_(value)
|
|
89
|
-
except (ValueError, TypeError):
|
|
90
|
-
raise BadParameter(
|
|
91
|
-
f"Invalid value for {param_name}: {value} (expected {type_.__name__})"
|
|
92
|
-
)
|
|
93
|
-
|
|
94
|
-
def validate_choice(
|
|
95
|
-
value: Any,
|
|
96
|
-
choices: List[Any],
|
|
97
|
-
param_name: str,
|
|
98
|
-
case_sensitive: bool = True
|
|
99
|
-
) -> None:
|
|
100
|
-
"""
|
|
101
|
-
Validate value is one of allowed choices.
|
|
102
|
-
|
|
103
|
-
Args:
|
|
104
|
-
value: Value to validate
|
|
105
|
-
choices: List of allowed choices
|
|
106
|
-
param_name: Parameter name for error messages
|
|
107
|
-
case_sensitive: Whether to do case-sensitive comparison
|
|
108
|
-
|
|
109
|
-
Raises:
|
|
110
|
-
BadParameter: If value not in choices
|
|
111
|
-
"""
|
|
112
|
-
if not case_sensitive and isinstance(value, str):
|
|
113
|
-
if value.lower() not in [str(c).lower() for c in choices]:
|
|
114
|
-
raise BadParameter(
|
|
115
|
-
f"Invalid choice for {param_name}: {value} "
|
|
116
|
-
f"(choose from {', '.join(str(c) for c in choices)})"
|
|
117
|
-
)
|
|
118
|
-
elif value not in choices:
|
|
119
|
-
raise BadParameter(
|
|
120
|
-
f"Invalid choice for {param_name}: {value} "
|
|
121
|
-
f"(choose from {', '.join(str(c) for c in choices)})"
|
|
122
|
-
)
|
|
123
|
-
|
|
124
|
-
def load_config_file(
|
|
125
|
-
path: Union[str, Path],
|
|
126
|
-
format: str = 'auto',
|
|
127
|
-
required: bool = True
|
|
128
|
-
) -> Dict[str, Any]:
|
|
129
|
-
"""
|
|
130
|
-
Load configuration from file.
|
|
131
|
-
|
|
132
|
-
Args:
|
|
133
|
-
path: Path to config file
|
|
134
|
-
format: File format (json, yaml, or auto)
|
|
135
|
-
required: Whether file is required
|
|
136
|
-
|
|
137
|
-
Returns:
|
|
138
|
-
Configuration dictionary
|
|
139
|
-
|
|
140
|
-
Raises:
|
|
141
|
-
UsageError: If required file not found or invalid format
|
|
142
|
-
"""
|
|
143
|
-
path = Path(os.path.expanduser(path))
|
|
144
|
-
|
|
145
|
-
if not path.exists():
|
|
146
|
-
if required:
|
|
147
|
-
raise UsageError(f"Config file not found: {path}")
|
|
148
|
-
return {}
|
|
149
|
-
|
|
150
|
-
# Auto-detect format from extension
|
|
151
|
-
if format == 'auto':
|
|
152
|
-
format = path.suffix.lstrip('.').lower()
|
|
153
|
-
if format not in ('json', 'yaml', 'yml'):
|
|
154
|
-
raise UsageError(f"Unsupported config format: {format}")
|
|
155
|
-
|
|
156
|
-
try:
|
|
157
|
-
with open(path) as f:
|
|
158
|
-
if format == 'json':
|
|
159
|
-
return json.load(f)
|
|
160
|
-
elif format in ('yaml', 'yml'):
|
|
161
|
-
return yaml.safe_load(f)
|
|
162
|
-
else:
|
|
163
|
-
raise UsageError(f"Unsupported config format: {format}")
|
|
164
|
-
except Exception as e:
|
|
165
|
-
raise UsageError(f"Error loading config file: {str(e)}")
|
|
166
|
-
|
|
167
|
-
def parse_key_value(
|
|
168
|
-
value: str,
|
|
169
|
-
separator: str = '='
|
|
170
|
-
) -> tuple:
|
|
171
|
-
"""
|
|
172
|
-
Parse key-value string.
|
|
173
|
-
|
|
174
|
-
Args:
|
|
175
|
-
value: String in format "key=value"
|
|
176
|
-
separator: Key-value separator
|
|
177
|
-
|
|
178
|
-
Returns:
|
|
179
|
-
Tuple of (key, value)
|
|
180
|
-
|
|
181
|
-
Raises:
|
|
182
|
-
BadParameter: If string not in key=value format
|
|
183
|
-
"""
|
|
184
|
-
try:
|
|
185
|
-
key, value = value.split(separator, 1)
|
|
186
|
-
return key.strip(), value.strip()
|
|
187
|
-
except ValueError:
|
|
188
|
-
raise BadParameter(
|
|
189
|
-
f"Invalid key-value pair: {value} (expected format: key{separator}value)"
|
|
190
|
-
)
|
|
191
|
-
|
|
192
|
-
def parse_list(
|
|
193
|
-
value: str,
|
|
194
|
-
separator: str = ','
|
|
195
|
-
) -> List[str]:
|
|
196
|
-
"""
|
|
197
|
-
Parse comma-separated list.
|
|
198
|
-
|
|
199
|
-
Args:
|
|
200
|
-
value: Comma-separated string
|
|
201
|
-
separator: List item separator
|
|
202
|
-
|
|
203
|
-
Returns:
|
|
204
|
-
List of strings
|
|
205
|
-
"""
|
|
206
|
-
return [x.strip() for x in value.split(separator) if x.strip()]
|
|
207
|
-
|
|
208
|
-
def parse_dict(
|
|
209
|
-
value: str,
|
|
210
|
-
item_separator: str = ',',
|
|
211
|
-
key_value_separator: str = '='
|
|
212
|
-
) -> Dict[str, str]:
|
|
213
|
-
"""
|
|
214
|
-
Parse dictionary string.
|
|
215
|
-
|
|
216
|
-
Args:
|
|
217
|
-
value: String in format "key1=value1,key2=value2"
|
|
218
|
-
item_separator: Separator between items
|
|
219
|
-
key_value_separator: Separator between keys and values
|
|
220
|
-
|
|
221
|
-
Returns:
|
|
222
|
-
Dictionary of key-value pairs
|
|
223
|
-
|
|
224
|
-
Example:
|
|
225
|
-
>>> parse_dict("name=test,count=42")
|
|
226
|
-
{'name': 'test', 'count': '42'}
|
|
227
|
-
"""
|
|
228
|
-
result = {}
|
|
229
|
-
if not value:
|
|
230
|
-
return result
|
|
231
|
-
|
|
232
|
-
items = parse_list(value, item_separator)
|
|
233
|
-
for item in items:
|
|
234
|
-
key, value = parse_key_value(item, key_value_separator)
|
|
235
|
-
result[key] = value
|
|
236
|
-
|
|
237
|
-
return result
|
|
238
|
-
|
|
239
|
-
def get_env_var(
|
|
240
|
-
name: str,
|
|
241
|
-
type_: Type = str,
|
|
242
|
-
required: bool = False,
|
|
243
|
-
default: Any = None
|
|
244
|
-
) -> Any:
|
|
245
|
-
"""
|
|
246
|
-
Get and validate environment variable.
|
|
247
|
-
|
|
248
|
-
Args:
|
|
249
|
-
name: Environment variable name
|
|
250
|
-
type_: Expected type
|
|
251
|
-
required: Whether variable is required
|
|
252
|
-
default: Default value if not set
|
|
253
|
-
|
|
254
|
-
Returns:
|
|
255
|
-
Environment variable value
|
|
256
|
-
|
|
257
|
-
Raises:
|
|
258
|
-
UsageError: If required variable not set
|
|
259
|
-
"""
|
|
260
|
-
value = os.environ.get(name)
|
|
261
|
-
|
|
262
|
-
if value is None:
|
|
263
|
-
if required:
|
|
264
|
-
raise UsageError(f"Required environment variable not set: {name}")
|
|
265
|
-
return default
|
|
266
|
-
|
|
267
|
-
return convert_type(value, type_, name)
|
|
1
|
+
"""Utility functions for parsing and validating command-line arguments."""
|
|
2
|
+
|
|
3
|
+
import os
|
|
4
|
+
import json
|
|
5
|
+
import yaml
|
|
6
|
+
from pathlib import Path
|
|
7
|
+
from typing import Any, Dict, List, Optional, Union, Type
|
|
8
|
+
|
|
9
|
+
from ..exceptions import BadParameter, UsageError
|
|
10
|
+
|
|
11
|
+
def parse_args(args: List[str]) -> Dict[str, Any]:
|
|
12
|
+
"""
|
|
13
|
+
Parse command line arguments into a dictionary.
|
|
14
|
+
|
|
15
|
+
Args:
|
|
16
|
+
args: List of command line arguments
|
|
17
|
+
|
|
18
|
+
Returns:
|
|
19
|
+
Dictionary of parsed arguments
|
|
20
|
+
|
|
21
|
+
Example:
|
|
22
|
+
>>> parse_args(['--name', 'test', '--flag', '-n', '42'])
|
|
23
|
+
{'name': 'test', 'flag': True, 'n': '42'}
|
|
24
|
+
"""
|
|
25
|
+
parsed = {}
|
|
26
|
+
i = 0
|
|
27
|
+
while i < len(args):
|
|
28
|
+
arg = args[i]
|
|
29
|
+
|
|
30
|
+
# Handle flags/options
|
|
31
|
+
if arg.startswith('-'):
|
|
32
|
+
key = arg.lstrip('-').replace('-', '_')
|
|
33
|
+
|
|
34
|
+
# Check if next arg is a value or another flag
|
|
35
|
+
if i + 1 >= len(args) or args[i + 1].startswith('-'):
|
|
36
|
+
parsed[key] = True # Flag without value
|
|
37
|
+
else:
|
|
38
|
+
parsed[key] = args[i + 1]
|
|
39
|
+
i += 1
|
|
40
|
+
else:
|
|
41
|
+
# Positional argument
|
|
42
|
+
parsed[f'arg{len([k for k in parsed.keys() if k.startswith("arg")])}'] = arg
|
|
43
|
+
|
|
44
|
+
i += 1
|
|
45
|
+
|
|
46
|
+
return parsed
|
|
47
|
+
|
|
48
|
+
def validate_required(
|
|
49
|
+
params: Dict[str, Any],
|
|
50
|
+
required: List[str]
|
|
51
|
+
) -> None:
|
|
52
|
+
"""
|
|
53
|
+
Validate required parameters are present.
|
|
54
|
+
|
|
55
|
+
Args:
|
|
56
|
+
params: Parameter dictionary
|
|
57
|
+
required: List of required parameter names
|
|
58
|
+
|
|
59
|
+
Raises:
|
|
60
|
+
UsageError: If required parameter is missing
|
|
61
|
+
"""
|
|
62
|
+
missing = [p for p in required if p not in params]
|
|
63
|
+
if missing:
|
|
64
|
+
raise UsageError(f"Missing required parameters: {', '.join(missing)}")
|
|
65
|
+
|
|
66
|
+
def convert_type(
|
|
67
|
+
value: str,
|
|
68
|
+
type_: Type,
|
|
69
|
+
param_name: str
|
|
70
|
+
) -> Any:
|
|
71
|
+
"""
|
|
72
|
+
Convert string value to specified type.
|
|
73
|
+
|
|
74
|
+
Args:
|
|
75
|
+
value: String value to convert
|
|
76
|
+
type_: Target type
|
|
77
|
+
param_name: Parameter name for error messages
|
|
78
|
+
|
|
79
|
+
Returns:
|
|
80
|
+
Converted value
|
|
81
|
+
|
|
82
|
+
Raises:
|
|
83
|
+
BadParameter: If conversion fails
|
|
84
|
+
"""
|
|
85
|
+
try:
|
|
86
|
+
if type_ == bool:
|
|
87
|
+
return value.lower() in ('true', 't', 'yes', 'y', '1')
|
|
88
|
+
return type_(value)
|
|
89
|
+
except (ValueError, TypeError):
|
|
90
|
+
raise BadParameter(
|
|
91
|
+
f"Invalid value for {param_name}: {value} (expected {type_.__name__})"
|
|
92
|
+
)
|
|
93
|
+
|
|
94
|
+
def validate_choice(
|
|
95
|
+
value: Any,
|
|
96
|
+
choices: List[Any],
|
|
97
|
+
param_name: str,
|
|
98
|
+
case_sensitive: bool = True
|
|
99
|
+
) -> None:
|
|
100
|
+
"""
|
|
101
|
+
Validate value is one of allowed choices.
|
|
102
|
+
|
|
103
|
+
Args:
|
|
104
|
+
value: Value to validate
|
|
105
|
+
choices: List of allowed choices
|
|
106
|
+
param_name: Parameter name for error messages
|
|
107
|
+
case_sensitive: Whether to do case-sensitive comparison
|
|
108
|
+
|
|
109
|
+
Raises:
|
|
110
|
+
BadParameter: If value not in choices
|
|
111
|
+
"""
|
|
112
|
+
if not case_sensitive and isinstance(value, str):
|
|
113
|
+
if value.lower() not in [str(c).lower() for c in choices]:
|
|
114
|
+
raise BadParameter(
|
|
115
|
+
f"Invalid choice for {param_name}: {value} "
|
|
116
|
+
f"(choose from {', '.join(str(c) for c in choices)})"
|
|
117
|
+
)
|
|
118
|
+
elif value not in choices:
|
|
119
|
+
raise BadParameter(
|
|
120
|
+
f"Invalid choice for {param_name}: {value} "
|
|
121
|
+
f"(choose from {', '.join(str(c) for c in choices)})"
|
|
122
|
+
)
|
|
123
|
+
|
|
124
|
+
def load_config_file(
|
|
125
|
+
path: Union[str, Path],
|
|
126
|
+
format: str = 'auto',
|
|
127
|
+
required: bool = True
|
|
128
|
+
) -> Dict[str, Any]:
|
|
129
|
+
"""
|
|
130
|
+
Load configuration from file.
|
|
131
|
+
|
|
132
|
+
Args:
|
|
133
|
+
path: Path to config file
|
|
134
|
+
format: File format (json, yaml, or auto)
|
|
135
|
+
required: Whether file is required
|
|
136
|
+
|
|
137
|
+
Returns:
|
|
138
|
+
Configuration dictionary
|
|
139
|
+
|
|
140
|
+
Raises:
|
|
141
|
+
UsageError: If required file not found or invalid format
|
|
142
|
+
"""
|
|
143
|
+
path = Path(os.path.expanduser(path))
|
|
144
|
+
|
|
145
|
+
if not path.exists():
|
|
146
|
+
if required:
|
|
147
|
+
raise UsageError(f"Config file not found: {path}")
|
|
148
|
+
return {}
|
|
149
|
+
|
|
150
|
+
# Auto-detect format from extension
|
|
151
|
+
if format == 'auto':
|
|
152
|
+
format = path.suffix.lstrip('.').lower()
|
|
153
|
+
if format not in ('json', 'yaml', 'yml'):
|
|
154
|
+
raise UsageError(f"Unsupported config format: {format}")
|
|
155
|
+
|
|
156
|
+
try:
|
|
157
|
+
with open(path) as f:
|
|
158
|
+
if format == 'json':
|
|
159
|
+
return json.load(f)
|
|
160
|
+
elif format in ('yaml', 'yml'):
|
|
161
|
+
return yaml.safe_load(f)
|
|
162
|
+
else:
|
|
163
|
+
raise UsageError(f"Unsupported config format: {format}")
|
|
164
|
+
except Exception as e:
|
|
165
|
+
raise UsageError(f"Error loading config file: {str(e)}")
|
|
166
|
+
|
|
167
|
+
def parse_key_value(
|
|
168
|
+
value: str,
|
|
169
|
+
separator: str = '='
|
|
170
|
+
) -> tuple:
|
|
171
|
+
"""
|
|
172
|
+
Parse key-value string.
|
|
173
|
+
|
|
174
|
+
Args:
|
|
175
|
+
value: String in format "key=value"
|
|
176
|
+
separator: Key-value separator
|
|
177
|
+
|
|
178
|
+
Returns:
|
|
179
|
+
Tuple of (key, value)
|
|
180
|
+
|
|
181
|
+
Raises:
|
|
182
|
+
BadParameter: If string not in key=value format
|
|
183
|
+
"""
|
|
184
|
+
try:
|
|
185
|
+
key, value = value.split(separator, 1)
|
|
186
|
+
return key.strip(), value.strip()
|
|
187
|
+
except ValueError:
|
|
188
|
+
raise BadParameter(
|
|
189
|
+
f"Invalid key-value pair: {value} (expected format: key{separator}value)"
|
|
190
|
+
)
|
|
191
|
+
|
|
192
|
+
def parse_list(
|
|
193
|
+
value: str,
|
|
194
|
+
separator: str = ','
|
|
195
|
+
) -> List[str]:
|
|
196
|
+
"""
|
|
197
|
+
Parse comma-separated list.
|
|
198
|
+
|
|
199
|
+
Args:
|
|
200
|
+
value: Comma-separated string
|
|
201
|
+
separator: List item separator
|
|
202
|
+
|
|
203
|
+
Returns:
|
|
204
|
+
List of strings
|
|
205
|
+
"""
|
|
206
|
+
return [x.strip() for x in value.split(separator) if x.strip()]
|
|
207
|
+
|
|
208
|
+
def parse_dict(
|
|
209
|
+
value: str,
|
|
210
|
+
item_separator: str = ',',
|
|
211
|
+
key_value_separator: str = '='
|
|
212
|
+
) -> Dict[str, str]:
|
|
213
|
+
"""
|
|
214
|
+
Parse dictionary string.
|
|
215
|
+
|
|
216
|
+
Args:
|
|
217
|
+
value: String in format "key1=value1,key2=value2"
|
|
218
|
+
item_separator: Separator between items
|
|
219
|
+
key_value_separator: Separator between keys and values
|
|
220
|
+
|
|
221
|
+
Returns:
|
|
222
|
+
Dictionary of key-value pairs
|
|
223
|
+
|
|
224
|
+
Example:
|
|
225
|
+
>>> parse_dict("name=test,count=42")
|
|
226
|
+
{'name': 'test', 'count': '42'}
|
|
227
|
+
"""
|
|
228
|
+
result = {}
|
|
229
|
+
if not value:
|
|
230
|
+
return result
|
|
231
|
+
|
|
232
|
+
items = parse_list(value, item_separator)
|
|
233
|
+
for item in items:
|
|
234
|
+
key, value = parse_key_value(item, key_value_separator)
|
|
235
|
+
result[key] = value
|
|
236
|
+
|
|
237
|
+
return result
|
|
238
|
+
|
|
239
|
+
def get_env_var(
|
|
240
|
+
name: str,
|
|
241
|
+
type_: Type = str,
|
|
242
|
+
required: bool = False,
|
|
243
|
+
default: Any = None
|
|
244
|
+
) -> Any:
|
|
245
|
+
"""
|
|
246
|
+
Get and validate environment variable.
|
|
247
|
+
|
|
248
|
+
Args:
|
|
249
|
+
name: Environment variable name
|
|
250
|
+
type_: Expected type
|
|
251
|
+
required: Whether variable is required
|
|
252
|
+
default: Default value if not set
|
|
253
|
+
|
|
254
|
+
Returns:
|
|
255
|
+
Environment variable value
|
|
256
|
+
|
|
257
|
+
Raises:
|
|
258
|
+
UsageError: If required variable not set
|
|
259
|
+
"""
|
|
260
|
+
value = os.environ.get(name)
|
|
261
|
+
|
|
262
|
+
if value is None:
|
|
263
|
+
if required:
|
|
264
|
+
raise UsageError(f"Required environment variable not set: {name}")
|
|
265
|
+
return default
|
|
266
|
+
|
|
267
|
+
return convert_type(value, type_, name)
|