Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow downloading extra formats in the demo #617

Merged
merged 8 commits into from
Jan 14, 2025
Merged

Conversation

roedoejet
Copy link
Member

@roedoejet roedoejet commented Dec 20, 2024

PR Goal?

This PR allows us to download any of the available output formats through the demo (TextGrid, ReadAlong, Spectrogram, etc). We can also optional disable certain output formats

Fixes?

This adds to fixing #607

Feedback sought?

Test it out, confirm it works as expected.

Priority?

medium

Tests added?

We currently don't test the demo - but we're moving to do that in @joanise 's PR

How to test?

spin up a demo and try synthesizing other formats

Confidence?

medium-high (I've tested it and it works well on my Mac)

Version change?

Related PRs?

EveryVoiceTTS/FastSpeech2_lightning#105

Copy link

semanticdiff-com bot commented Dec 20, 2024

Review changes with  SemanticDiff

Changed Files
File Status
  everyvoice/cli.py  31% smaller
  everyvoice/demo/app.py  15% smaller
  everyvoice/model/feature_prediction/FastSpeech2_lightning  0% smaller
  everyvoice/tests/test_cli.py  0% smaller

Copy link

codecov bot commented Dec 20, 2024

Codecov Report

Attention: Patch coverage is 37.83784% with 23 lines in your changes missing coverage. Please review.

Project coverage is 78.66%. Comparing base (1c2aae6) to head (f84e604).
Report is 12 commits behind head on main.

Files with missing lines Patch % Lines
everyvoice/demo/app.py 32.35% 21 Missing and 2 partials ⚠️
Additional details and impacted files
@@            Coverage Diff             @@
##             main     #617      +/-   ##
==========================================
+ Coverage   76.75%   78.66%   +1.91%     
==========================================
  Files          46       47       +1     
  Lines        3446     3900     +454     
  Branches      470      575     +105     
==========================================
+ Hits         2645     3068     +423     
- Misses        700      723      +23     
- Partials      101      109       +8     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

Copy link
Contributor

github-actions bot commented Dec 20, 2024

CLI load time: 0:00.26
Pull Request HEAD: f84e60420c0e5aa837525dd8c86c4dde84072816
Imports that take more than 0.1 s:
import time: self [us] | cumulative | imported package
import time:      1080 |     104298 |     typer.main
import time:       282 |     123373 |   typer
import time:      8064 |     202657 | everyvoice.cli

Copy link
Member

@joanise joanise left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice work, though I have a number of comments and suggestions.

In general, I find it not very intuitive to have to re-synthesize if I change the output format. I guess from how things work, it's probably unavoidable, but it's not ideal UX. Maybe the File Output box could have a hint indicating that Output Format is where you change what's available for download here?

When you keep just the wav output format, it's confusing that the File Output box is present but you can't interact with it. I wanted to download my audio file, and I tried to find it in the File Output box. It took me a little while to locate the download button in the Audio box above.

@@ -616,6 +616,12 @@ def demo(
"-s",
help="Specify speakers to be included in the demo. Example: everyvoice demo <path_to_text_to_spec_model> <path_to_spec_to_wav_model> --speaker speaker_1 --speaker Sue",
),
outputs: List[str] = typer.Option(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It would be helpful to enumerate the valid options in the help message.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ditto for --accelerator, while I'm thinking about it... And for --language and --speaker we should state that they have to be language(s) and speaker(s) known to the model.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For this PR, please address listing valid values for --output-format, fixing the other help messages is gravy and could go into a separate PR or issue.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ditto for --accelerator, while I'm thinking about it... And for --language and --speaker we should state that they have to be language(s) and speaker(s) known to the model.

I think they do get listed, don't they? Like if you type a speaker that doesn't exist, I thought the error message listed out all the possible speakers. The output formats are dependent on the version of everyvoice installed, so we could include that in the help message, but the language and speaker are model-dependent, so we wouldn't be able to include the lists of those in the help message, just in the error message.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What I mean is that the everyvoice demo -h message should say something like "valid values are the language(s) and speaker(s) the model was trained on", or something to that effect, maybe more concisely. As the documentation stands, if you're not familiar with things yet, it's a bit mysterious how you're supposed to know what values you can use there.
And I know if you're just trained things, it's going to be obvious, but the point the of the help message is to support you when the information is not already obvious to you.

Comment on lines 305 to 308
else:
print(
f"Attention: This model is not able to produce '{output}' as an output. The '{output}' option will not be available for selection. Please choose from the following possible outputs: {', '.join(possible_outputs)}"
)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This needs to be a fatal error with an immediate exit, and the message is misleading: it's not that the model can't produce the requested output, it's that the software has no implementation for it. Right now, everyvoice demo -O foo fs2.ckpt voc.ckpt prints this message about foo and then continues anyway and crashes with an exception a few lines later.

This is really CLI error checking, it should happen much earlier in this function, in particular before we load any checkpoint, so the error is dumped right away without having to wait 20 seconds or more for models to load first. You might get all this for nearly free if you define the list of valid values for outputs in cli.py's demo() function as I already suggested elsewhere.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

BTW, the RAS output specifiers are readalong_xml and readalong_html with an underscore instead of a hyphen like in everyvoice synthesize from-text. They should be unified, using hyphens here too.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

agreed!

wav_output = wav_writer.get_filename(basename, speaker, language)
file_writer = None
file_output = None
if output_format == SynthesizeOutputFormats.readalong_html.name:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This and all the other callback constructor calls really ought to be able to be replaced by a single call to get_synthesis_output_callbacks, no?

I realize you need the wav writer to create the RAS_html writer, but that's already done too.

Refactoring suggestion: have get_synthesis_output_callbacks return a dict where the key are the output types, and the values are the writers. Then here you can use writers["wav"] and writers[output_format] to access the two writers you need, having passed ["wav", output_format] (or the proper Enum form if need be) as the output_type argument to get_synthesis_output_callbacks.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yea, this is a good idea. I would definitely be in favour of this refactor, which I think would clean up some of the acrobatics we're currently doing in order to get the wav writer separately.

@roedoejet
Copy link
Member Author

In general, I find it not very intuitive to have to re-synthesize if I change the output format. I guess from how things work, it's probably unavoidable, but it's not ideal UX. Maybe the File Output box could have a hint indicating that Output Format is where you change what's available for download here?

I agree, but I'm not sure gradio provides a callback for when the output format gets changed. It's possible, but given how expensive it is to synthesize (depending on the underlying hardware), I'm not super opposed to this UX, despite agreeing that it's clunky, as you say.

When you keep just the wav output format, it's confusing that the File Output box is present but you can't interact with it. I wanted to download my audio file, and I tried to find it in the File Output box. It took me a little while to locate the download button in the Audio box above.

Yes, the problem is that I don't believe gradio allows you to dynamically change the interface once it's been rendered. So we might be stuck with that. Note that it doesn't get rendered if the only possible output format is "wav" (i.e. if the demo command disables all the other output formats).

@marctessier
Copy link
Collaborator

Having issues trying to generate a spec. The other formats being generated are good and valid :-)

Also, The demo server is generated this error traceback at times. In this example below, The first time I try to synth / output a "spec" , no messages or "traceback" in logs... ( The web interface did show "Error").
The second exact transaction / same test generated the error message.

2025-01-09 10:33:02.968 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:230 - Saving pytorch output to synthesis_output/synthesized_spec
GPU available: False, used: False
TPU available: False, using: 0 TPU cores
HPU available: False, using: 0 HPUs

Processing text ['This is a new test ']
Predicting DataLoader 0: 100%|████████████████████████████████████████████████████████████████████| 1/1 [00:01<00:00,  0.84it/s]2025-01-09 10:33:04.825 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:518 - Saving wav output to synthesis_output/wav
2025-01-09 10:33:06.997 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:230 - Saving pytorch output to synthesis_output/synthesized_spec



*******  Second  exact same test below a couple  seconds later but this time it generated a traceback log.


2025-01-09 10:33:14.391 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:230 - Saving pytorch output to synthesis_output/synthesized_spec
GPU available: False, used: False
TPU available: False, using: 0 TPU cores
HPU available: False, using: 0 HPUs

Traceback (most recent call last):
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/gradio/queueing.py", line 625, in process_events
    response = await route_utils.call_process_api(
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/gradio/route_utils.py", line 322, in call_process_api
    output = await app.get_blocks().process_api(
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/gradio/blocks.py", line 2045, in process_api
    result = await self.call_function(
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/gradio/blocks.py", line 1592, in call_function
    prediction = await anyio.to_thread.run_sync(  # type: ignore
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/anyio/to_thread.py", line 56, in run_sync
    return await get_async_backend().run_sync_in_worker_thread(
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/anyio/_backends/_asyncio.py", line 2461, in run_sync_in_worker_thread
    return await future
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/anyio/_backends/_asyncio.py", line 962, in run
    result = context.run(func, *args)
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/miniforge3/envs/EveryVoice_pr617/lib/python3.10/site-packages/gradio/utils.py", line 870, in wrapper
    response = f(*args, **kwargs)
  File "/gpfs/fs5/nrc/nrc-fs1/ict/others/u/tes001/TxT2SPEECH/EveryVoice_pr617/everyvoice/demo/app.py", line 171, in synthesize_audio
    file_writer.save_aligned_text_to_file(
AttributeError: 'PredictionWritingSpecCallback' object has no attribute 'save_aligned_text_to_file'
Processing text ['This is a new test ']
Predicting DataLoader 0: 100%|████████████████████████████████████████████████████████████████████| 1/1 [00:01<00:00,  0.83it/s]2025-01-09 10:33:16.273 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:518 - Saving wav output to synthesis_output/wav
2025-01-09 10:33:19.504 | INFO     | everyvoice.model.feature_prediction.FastSpeech2_lightning.fs2.prediction_writing_callback:__init__:230 - Saving pytorch output to synthesis_output/synthesized_spec

Below is the screen capture of the message presented to the end user from the Gradio web interface when using select output format "spec" after "synth".

Screenshot 2025-01-09 at 10 23 29

@joanise
Copy link
Member

joanise commented Jan 9, 2025

Hi @marctessier I fixed the problem with the Spec in #619. Please test with that PR's code instead of this one.

Details:
 - everyvoice demo -h will now list the valid output formats
 - code is DRYer: we only instantiate the callbacks in synthesize_helper, which
   returns them so we can access them to find out the output filenames where we
   need them
 - the File Output box is hidden when wav is the only output format
Plus minimal unit testing of this updated error checking code.
@roedoejet roedoejet merged commit 2a111c0 into main Jan 14, 2025
5 of 6 checks passed
@roedoejet roedoejet deleted the dev.ap/demo-formats branch January 14, 2025 17:15
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants