add validate.timecode_format

add None checks for callbacks with optional values
This commit is contained in:
onyx-and-iris 2026-01-10 13:59:55 +00:00
parent 5c7fc24839
commit 8aa1fb2c09
3 changed files with 37 additions and 4 deletions

View File

@ -31,7 +31,7 @@ def list_(
] = None, ] = None,
): ):
"""List groups in a scene.""" """List groups in a scene."""
if not scene_name: if scene_name is None:
scene_name = ctx.obj['obsws'].get_current_program_scene().scene_name scene_name = ctx.obj['obsws'].get_current_program_scene().scene_name
resp = ctx.obj['obsws'].get_scene_item_list(scene_name) resp = ctx.obj['obsws'].get_scene_item_list(scene_name)

View File

@ -4,7 +4,7 @@ from typing import Annotated, Optional
import typer import typer
from . import console, util from . import console, util, validate
from .alias import SubTyperAliasGroup from .alias import SubTyperAliasGroup
app = typer.Typer(cls=SubTyperAliasGroup) app = typer.Typer(cls=SubTyperAliasGroup)
@ -24,7 +24,9 @@ def cursor(
timecode: Annotated[ timecode: Annotated[
Optional[str], Optional[str],
typer.Argument( typer.Argument(
..., help='The timecode to set the cursor to (format: HH:MM:SS).' ...,
help='The timecode to set the cursor to (format: HH:MM:SS).',
callback=validate.timecode_format,
), ),
] = None, ] = None,
): ):

View File

@ -1,5 +1,7 @@
"""module containing validation functions.""" """module containing validation functions."""
from typing import Optional
import typer import typer
from . import console from . import console
@ -26,8 +28,11 @@ def input_not_in_inputs(ctx: typer.Context, input_name: str) -> bool:
return input_name return input_name
def scene_in_scenes(ctx: typer.Context, scene_name: str) -> str: def scene_in_scenes(ctx: typer.Context, scene_name: Optional[str]) -> str | None:
"""Check if a scene exists in the list of scenes.""" """Check if a scene exists in the list of scenes."""
if scene_name is None:
return scene_name
resp = ctx.obj['obsws'].get_scene_list() resp = ctx.obj['obsws'].get_scene_list()
if not any(scene.get('sceneName') == scene_name for scene in resp.scenes): if not any(scene.get('sceneName') == scene_name for scene in resp.scenes):
console.err.print(f'Scene [yellow]{scene_name}[/yellow] not found.') console.err.print(f'Scene [yellow]{scene_name}[/yellow] not found.')
@ -104,3 +109,29 @@ def kind_in_input_kinds(ctx: typer.Context, input_kind: str) -> str:
console.err.print(f'Input kind [yellow]{input_kind}[/yellow] not found.') console.err.print(f'Input kind [yellow]{input_kind}[/yellow] not found.')
raise typer.Exit(1) raise typer.Exit(1)
return input_kind return input_kind
def timecode_format(ctx: typer.Context, timecode: Optional[str]) -> str | None:
"""Validate that a timecode is in HH:MM:SS or MM:SS format."""
if timecode is None:
return timecode
match timecode.split(':'):
case [mm, ss]:
if not (mm.isdigit() and ss.isdigit()):
console.err.print(
f'Timecode [yellow]{timecode}[/yellow] is not valid. Use MM:SS or HH:MM:SS format.'
)
raise typer.Exit(1)
case [hh, mm, ss]:
if not (hh.isdigit() and mm.isdigit() and ss.isdigit()):
console.err.print(
f'Timecode [yellow]{timecode}[/yellow] is not valid. Use MM:SS or HH:MM:SS format.'
)
raise typer.Exit(1)
case _:
console.err.print(
f'Timecode [yellow]{timecode}[/yellow] is not valid. Use MM:SS or HH:MM:SS format.'
)
raise typer.Exit(1)
return timecode