Spaces:
Runtime error
Runtime error
File size: 11,033 Bytes
1a942eb |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 |
"""
Module defining common utility functions and classes for the
web application of the Ultimate RVC project.
"""
from typing import Any, Concatenate
from collections.abc import Callable, Sequence
import gradio as gr
from ultimate_rvc.core.exceptions import NotProvidedError
from ultimate_rvc.core.generate.song_cover import (
get_named_song_dirs,
get_song_cover_name,
)
from ultimate_rvc.core.manage.audio import get_saved_output_audio
from ultimate_rvc.web.typing_extra import (
ComponentVisibilityKwArgs,
DropdownChoices,
DropdownValue,
TextBoxKwArgs,
UpdateDropdownKwArgs,
)
PROGRESS_BAR = gr.Progress()
def exception_harness[T, **P](
fn: Callable[P, T],
info_msg: str | None = None,
) -> Callable[P, T]:
"""
Wrap a function in a harness that catches exceptions and re-raises
them as instances of `gradio.Error`.
Parameters
----------
fn : Callable[P, T]
The function to wrap.
info_msg : str, optional
Message to display in an info-box pop-up after the function
executes successfully.
Returns
-------
Callable[P, T]
The wrapped function.
"""
def _wrapped_fn(*args: P.args, **kwargs: P.kwargs) -> T:
try:
res = fn(*args, **kwargs)
except gr.Error:
raise
except NotProvidedError as e:
msg = e.ui_msg or e
raise gr.Error(str(msg)) from None
except Exception as e:
raise gr.Error(str(e)) from e
else:
if info_msg:
gr.Info(info_msg, duration=0.5)
return res
return _wrapped_fn
def confirmation_harness[T, **P](
fn: Callable[P, T],
) -> Callable[Concatenate[bool, P], T]:
"""
Wrap a function in a harness that requires a confirmation before
executing and catches exceptions, re-raising them as instances of
`gradio.Error`.
Parameters
----------
fn : Callable[P, T]
The function to wrap.
Returns
-------
Callable[Concatenate[bool, P], T]
The wrapped function.
"""
def _wrapped_fn(confirm: bool, *args: P.args, **kwargs: P.kwargs) -> T:
if confirm:
return exception_harness(fn)(*args, **kwargs)
err_msg = "Confirmation missing!"
raise gr.Error(err_msg)
return _wrapped_fn
def render_msg(
template: str,
*args: str,
display_info: bool = False,
**kwargs: str,
) -> str:
"""
Render a message template with the provided arguments.
Parameters
----------
template : str
Message template to render.
args : str
Positional arguments to pass to the template.
display_info : bool, default=False
Whether to display the rendered message as an info message
in addition to returning it.
kwargs : str
Keyword arguments to pass to the template.
Returns
-------
str
Rendered message.
"""
msg = template.format(*args, **kwargs)
if display_info:
gr.Info(msg)
return msg
def confirm_box_js(msg: str) -> str:
"""
Generate a JavaScript code snippet which:
* defines an anonymous function that takes one named parameter and
zero or more unnamed parameters
* renders a confirmation box
* returns the choice selected by the user in that confirmation
box in addition to any unnamed parameters passed to the function.
Parameters
----------
msg : str
Message to display in the confirmation box rendered by the
JavaScript code snippet.
Returns
-------
str
The JavaScript code snippet.
"""
return f"(x, ...args) => [confirm('{msg}'), ...args]"
def update_value(x: str) -> dict[str, Any]:
"""
Update the value of a component.
Parameters
----------
x : str
New value for the component.
Returns
-------
dict[str, Any]
Dictionary which updates the value of the component.
"""
return gr.update(value=x)
def update_dropdowns[**P](
fn: Callable[P, DropdownChoices],
num_components: int,
value: DropdownValue = None,
value_indices: Sequence[int] = [],
*args: P.args,
**kwargs: P.kwargs,
) -> gr.Dropdown | tuple[gr.Dropdown, ...]:
"""
Update the choices and optionally the value of one or more dropdown
components.
Parameters
----------
fn : Callable[P, DropdownChoices]
Function to get updated choices for the dropdown components.
num_components : int
Number of dropdown components to update.
value : DropdownValue, optional
New value for dropdown components.
value_indices : Sequence[int], default=[]
Indices of dropdown components to update the value for.
args : P.args
Positional arguments to pass to the function used to update
dropdown choices.
kwargs : P.kwargs
Keyword arguments to pass to the function used to update
dropdown choices.
Returns
-------
gr.Dropdown | tuple[gr.Dropdown,...]
Updated dropdown component or components.
Raises
------
ValueError
If not all provided indices are unique or if an index exceeds
or is equal to the number of dropdown components.
"""
if len(value_indices) != len(set(value_indices)):
err_msg = "Value indices must be unique."
raise ValueError(err_msg)
if value_indices and max(value_indices) >= num_components:
err_msg = (
"Index of a dropdown component to update the value for exceeds the number"
" of dropdown components to update."
)
raise ValueError(err_msg)
updated_choices = fn(*args, **kwargs)
update_args_list: list[UpdateDropdownKwArgs] = [
{"choices": updated_choices} for _ in range(num_components)
]
for index in value_indices:
update_args_list[index]["value"] = value
match update_args_list:
case [update_args]:
# NOTE This is a workaround as gradio does not support
# singleton tuples for components.
return gr.Dropdown(**update_args)
case _:
return tuple(gr.Dropdown(**update_args) for update_args in update_args_list)
def update_cached_songs(
num_components: int,
value: DropdownValue = None,
value_indices: Sequence[int] = [],
) -> gr.Dropdown | tuple[gr.Dropdown, ...]:
"""
Update the choices of one or more dropdown components to the set of
currently cached songs.
Optionally update the default value of one or more of these
components.
Parameters
----------
num_components : int
Number of dropdown components to update.
value : DropdownValue, optional
New value for the dropdown components.
value_indices : Sequence[int], default=[]
Indices of dropdown components to update the value for.
Returns
-------
gr.Dropdown | tuple[gr.Dropdown,...]
Updated dropdown component or components.
"""
return update_dropdowns(get_named_song_dirs, num_components, value, value_indices)
def update_output_audio(
num_components: int,
value: DropdownValue = None,
value_indices: Sequence[int] = [],
) -> gr.Dropdown | tuple[gr.Dropdown, ...]:
"""
Update the choices of one or more dropdown components to the set of
currently saved output audio files.
Optionally update the default value of one or more of these
components.
Parameters
----------
num_components : int
Number of dropdown components to update.
value : DropdownValue, optional
New value for dropdown components.
value_indices : Sequence[int], default=[]
Indices of dropdown components to update the value for.
Returns
-------
gr.Dropdown | tuple[gr.Dropdown,...]
Updated dropdown component or components.
"""
return update_dropdowns(
get_saved_output_audio,
num_components,
value,
value_indices,
)
def toggle_visible_component(
num_components: int,
visible_index: int,
) -> dict[str, Any] | tuple[dict[str, Any], ...]:
"""
Reveal a single component from a set of components. All other
components are hidden.
Parameters
----------
num_components : int
Number of components to set visibility for.
visible_index : int
Index of the component to reveal.
Returns
-------
dict[str, Any] | tuple[dict[str, Any], ...]
A single dictionary or a tuple of dictionaries that update the
visibility of the components.
Raises
------
ValueError
If the visible index exceeds or is equal to the number of
components to set visibility for.
"""
if visible_index >= num_components:
err_msg = (
"Visible index must be less than the number of components to set visibility"
" for."
)
raise ValueError(err_msg)
update_args_list: list[ComponentVisibilityKwArgs] = [
{"visible": False, "value": None} for _ in range(num_components)
]
update_args_list[visible_index]["visible"] = True
match update_args_list:
case [update_args]:
return gr.update(**update_args)
case _:
return tuple(gr.update(**update_args) for update_args in update_args_list)
def update_song_cover_name(
effected_vocals_track: str | None = None,
song_dir: str | None = None,
model_name: str | None = None,
update_placeholder: bool = False,
) -> gr.Textbox:
"""
Update a textbox component so that it displays a suitable name for a
cover of a given song.
If the path of an existing song directory is provided, the name of
the song is inferred from that directory. If the name of a voice
model is not provided but the path of an existing song directory
and the path of an effected vocals track in that directory are
provided, then the voice model is inferred from the effected vocals
track.
Parameters
----------
effected_vocals_track : str, optional
The path to an effected vocals track.
song_dir : str, optional
The path to a song directory.
model_name : str, optional
The name of a voice model.
update_placeholder : bool, default=False
Whether to update the placeholder text instead of the value of
the textbox component.
Returns
-------
gr.Textbox
Textbox component with updated value or placeholder text.
"""
update_args: TextBoxKwArgs = {}
update_key = "placeholder" if update_placeholder else "value"
if effected_vocals_track or song_dir or model_name:
song_cover_name = get_song_cover_name(
effected_vocals_track,
song_dir,
model_name,
)
update_args[update_key] = song_cover_name
else:
update_args[update_key] = None
return gr.Textbox(**update_args)
|