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

Create test for a speech sample (InfiniteStreamRecognize) #9336

Open
wants to merge 12 commits into
base: main
Choose a base branch
from

Conversation

ssvir
Copy link
Contributor

@ssvir ssvir commented May 20, 2024

Description

Fixes #8261

Note: Before submitting a pull request, please open an issue for discussion if you are not associated with Google.

Checklist

  • I have followed Sample Format Guide
  • pom.xml parent set to latest shared-configuration
  • Appropriate changes to README are included in PR
  • These samples need a new API enabled in testing projects to pass (let us know which ones)
  • These samples need a new/updated env vars in testing projects set to pass (let us know which ones)
  • Tests pass: mvn clean verify required
  • Lint passes: mvn -P lint checkstyle:check required
  • Static Analysis: mvn -P lint clean compile pmd:cpd-check spotbugs:check advisory only
  • This sample adds a new sample directory, and I updated the CODEOWNERS file with the codeowners for this sample
  • This sample adds a new Product API, and I updated the Blunderbuss issue/PR auto-assigner with the codeowners for this sample
  • Please merge this PR for me once it is approved

@ssvir ssvir requested a review from Sita04 May 20, 2024 12:38
@ssvir ssvir requested review from yoshi-approver and a team as code owners May 20, 2024 12:38
@product-auto-label product-auto-label bot added samples Issues that are directly related to samples. api: speech Issues related to the Speech-to-Text API. labels May 20, 2024
@ssvir ssvir added kokoro:run Add this label to force Kokoro to re-run the tests. and removed api: speech Issues related to the Speech-to-Text API. samples Issues that are directly related to samples. labels May 20, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 20, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 21, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 21, 2024
@product-auto-label product-auto-label bot added the samples Issues that are directly related to samples. label May 21, 2024
@Sita04 Sita04 assigned minherz and unassigned Sita04 May 23, 2024
Comment on lines +52 to +53
InfiniteStreamRecognize.putDataToSharedQueue(readAudioFile());
InfiniteStreamRecognize.putDataToSharedQueue("exit".getBytes(StandardCharsets.UTF_8));
Copy link
Contributor

Choose a reason for hiding this comment

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

Are these lines a part of the test or a part of the setup?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

As a part of the test, readAudioFile(), we read the test file to recognise and in the next line, we command to stop recognition

Copy link
Contributor

Choose a reason for hiding this comment

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

Do you mean that the reading audio file and adding text to the already read binary audio data is a part of the test? What do these lines test?

Copy link
Contributor

@minherz minherz left a comment

Choose a reason for hiding this comment

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

Why this code sample does not have region tags?
This repository is intended for code samples that are used with documentation. Please, check for appropriate region tag.

@ssvir ssvir requested a review from minherz May 27, 2024 12:27
@ssvir ssvir added api: speech Issues related to the Speech-to-Text API. kokoro:run Add this label to force Kokoro to re-run the tests. labels May 27, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 27, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 28, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 28, 2024
@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label May 28, 2024
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label May 28, 2024
Copy link
Contributor

@minherz minherz left a comment

Choose a reason for hiding this comment

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

Thank you for addressing majority of items.
I still see that testing uses stdout buffer but I recognize that in this particular code sample other exit statuses are harder to implement.

I found a potential bug when calling "checking stop recognition flag" in event the response contains multiple items. Please, look into that.

Comment on lines +90 to +91
infiniteStreamingRecognize(options.langCode, micBuffer, sampleRate,
RecognitionConfig.AudioEncoding.LINEAR16);
Copy link
Contributor

Choose a reason for hiding this comment

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

I understand that this method exits when a stream has a word "exit".
It would be useful to document it in the code. Consider renaming variable to be more verabit (eg "exitRecognized") or write a comment.
nit: Otherwise, if a user ends up killing the app, a graceful termination on SIGKILL might be useful if resources require release.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I added a comment on how to stop, and this will also be printed when the application runs.

Copy link
Contributor

Choose a reason for hiding this comment

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

We have guidelines how code sample should look like. This repository is intended to host code used mainly for documentation code snippets. See go/code-snippets-101#what-is-a-code-snippet for the definition. It is not intended for demo applications or other complex solutions.
I agree that printing it makes more sense than place a comment. However, I see no code line that prints this instruction.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Could you please send a direct link, or if this link needs google account, could you please copy it here a phrase

Copy link
Contributor

Choose a reason for hiding this comment

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

The documentation is available for Googlers. This is why we experience challenges when external contributors introduce complex code samples or changes to them in this repo.

I would also like to better understand reasoning behind changing the code sample when the PR description explains changes as a test for the already existing code sample.

}

private static void checkStopRecognitionFlag(byte[] flag) {
if (flag.length < 10) {
Copy link
Contributor

Choose a reason for hiding this comment

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

nit: it would be nice to make more clear why 10 is selected as a threshold. consider using a constant with verbatim name or another method.

"%s: %s", convertMillisToDate(correctedTime), alternative.getTranscript());
lastTranscriptWasFinal = false;
}
checkStopRecognitionFlag(alternative.getTranscript().getBytes(StandardCharsets.UTF_8));
Copy link
Contributor

Choose a reason for hiding this comment

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

this method will fail if the "exit" word is not in the first element. can such thing happen? can the code reflect that?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

this code reflects only if the word "exit" will be present in a sentence , that's why we check element length before comparing

Copy link
Contributor

Choose a reason for hiding this comment

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

My question was related to line 256. Is it possible that other, less confident, alternatives interpret "exit" while the first one does not?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I think we have some possibility, so in this case, we should repeat "exit".

Copy link
Contributor

Choose a reason for hiding this comment

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

IMHO it does not look like a good code sample then. Code samples should be deterministic and easy to comprehend.

// wait responses from server
Thread.sleep(10000);
assertThat(bout.toString().toLowerCase()).contains("hi i want to");
assertThat(bout.toString().toLowerCase()).contains("recognition was stopped");
Copy link
Contributor

Choose a reason for hiding this comment

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

I listen to the commercial_mono.wav file and it does not contain a word "exit". Does it mean that onComplete() is called when the stream ends OR stopped?
I recommend to consider distinguishing these two events. Currently the impression from the code sample is that onComplete() is called when the recognition is stopped.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

As we have InfiniteStreamRecognize, I suppose we don't have some status-completed, so this code can call onComplete() only in one way when we stop the recognition

Copy link
Contributor

Choose a reason for hiding this comment

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

Integration tests are supposed to be deterministic, terminated, run minimal time required to execute the test and do a proper setup and clean up action so they aren't dependent on the state of the testing environment and do not consume resources beyond required.
From your comment it sounds like the test might not terminate, making it a subject for flakiness.

Please, reconsider the testing strategy because we prefer not to have a test at all to having a flaky test.

@ssvir ssvir added the kokoro:run Add this label to force Kokoro to re-run the tests. label Jun 5, 2024
@ssvir ssvir requested a review from minherz June 5, 2024 12:49
@kokoro-team kokoro-team removed the kokoro:run Add this label to force Kokoro to re-run the tests. label Jun 5, 2024
Copy link
Contributor

@minherz minherz left a comment

Choose a reason for hiding this comment

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

Please provide region tags for this code snippet.
Please follow guidelines from go/code-snippets-201#including-code-from-multiple-sections-of-a-file to exclude the main method and all auxiliary methods from the code snippet.
If there is no intention to use this sample as a code snippet, please consider using a different repository to host this code.

Comment on lines +90 to +91
infiniteStreamingRecognize(options.langCode, micBuffer, sampleRate,
RecognitionConfig.AudioEncoding.LINEAR16);
Copy link
Contributor

Choose a reason for hiding this comment

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

We have guidelines how code sample should look like. This repository is intended to host code used mainly for documentation code snippets. See go/code-snippets-101#what-is-a-code-snippet for the definition. It is not intended for demo applications or other complex solutions.
I agree that printing it makes more sense than place a comment. However, I see no code line that prints this instruction.

@@ -39,19 +40,16 @@
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.DataLine;
import javax.sound.sampled.DataLine.Info;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.TargetDataLine;

public class InfiniteStreamRecognize {

private static final int STREAMING_LIMIT = 290000; // ~5 minutes
Copy link
Contributor

Choose a reason for hiding this comment

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

nit: This code sample is called infinite streaming. The name "streaming limit" can be potentially confusing. Consider providing more meaningful name.


// Creating shared object
private static volatile BlockingQueue<byte[]> sharedQueue = new LinkedBlockingQueue<byte[]>();
private static TargetDataLine targetDataLine;
private static int BYTES_PER_BUFFER = 6400; // buffer size in bytes
Copy link
Contributor

Choose a reason for hiding this comment

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

nit: Consider selecting multiple of 2 for the buffer size as a general guideline. Is it possible to explain how to select a certain value?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I can't explain how to select a certain value.

Comment on lines 61 to 66
private static boolean newStream = true;
private static double bridgingOffset = 0;
private static boolean lastTranscriptWasFinal = false;
private static boolean stopRecognition = false;
private static StreamController referenceToStreamController;
private static ByteString tempByteString;
Copy link
Contributor

Choose a reason for hiding this comment

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

All constants in Java should use UPPER_SNAKE_CASE

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I don't have google acc :(

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@minherz By constant, do you mean the variable marked as final or static is constant too?

Copy link
Contributor

Choose a reason for hiding this comment

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

I don't have google acc :(

I looked into the case closer. It looks like you made changes to the code sample itself and not only added the test case. I will ask a Googler who opened the issue to review the changes you made to the code sample itself.

@minherz By constant, do you mean the variable marked as final or static is constant too?

I mean "constants" in Java. It should be literals declared with const or objects defined with final. Static or not static should not play a role.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yeap, I refactored this sample to have a possibility to replace the voice recognition thread with a file or another source of an audio byte stream

"%s: %s", convertMillisToDate(correctedTime), alternative.getTranscript());
lastTranscriptWasFinal = false;
}
checkStopRecognitionFlag(alternative.getTranscript().getBytes(StandardCharsets.UTF_8));
Copy link
Contributor

Choose a reason for hiding this comment

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

My question was related to line 256. Is it possible that other, less confident, alternatives interpret "exit" while the first one does not?

@ssvir ssvir requested a review from minherz June 11, 2024 12:35
@minherz minherz requested a review from anguillanneuf June 12, 2024 20:35
Copy link
Contributor

@minherz minherz left a comment

Choose a reason for hiding this comment

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

It is unclear whether the proposed testing strategy is deterministic. Loading audio file bytes and then adding text string to the same stream raises concerns about test performance.
With the current open questions and lack of access to the internal information regarding best practices for code samples I prefer to hold this change for now.
@Sita04 please have a look

Comment on lines +90 to +91
infiniteStreamingRecognize(options.langCode, micBuffer, sampleRate,
RecognitionConfig.AudioEncoding.LINEAR16);
Copy link
Contributor

Choose a reason for hiding this comment

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

The documentation is available for Googlers. This is why we experience challenges when external contributors introduce complex code samples or changes to them in this repo.

I would also like to better understand reasoning behind changing the code sample when the PR description explains changes as a test for the already existing code sample.

"%s: %s", convertMillisToDate(correctedTime), alternative.getTranscript());
lastTranscriptWasFinal = false;
}
checkStopRecognitionFlag(alternative.getTranscript().getBytes(StandardCharsets.UTF_8));
Copy link
Contributor

Choose a reason for hiding this comment

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

IMHO it does not look like a good code sample then. Code samples should be deterministic and easy to comprehend.

// wait responses from server
Thread.sleep(10000);
assertThat(bout.toString().toLowerCase()).contains("hi i want to");
assertThat(bout.toString().toLowerCase()).contains("recognition was stopped");
Copy link
Contributor

Choose a reason for hiding this comment

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

Integration tests are supposed to be deterministic, terminated, run minimal time required to execute the test and do a proper setup and clean up action so they aren't dependent on the state of the testing environment and do not consume resources beyond required.
From your comment it sounds like the test might not terminate, making it a subject for flakiness.

Please, reconsider the testing strategy because we prefer not to have a test at all to having a flaky test.

Comment on lines +52 to +53
InfiniteStreamRecognize.putDataToSharedQueue(readAudioFile());
InfiniteStreamRecognize.putDataToSharedQueue("exit".getBytes(StandardCharsets.UTF_8));
Copy link
Contributor

Choose a reason for hiding this comment

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

Do you mean that the reading audio file and adding text to the already read binary audio data is a part of the test? What do these lines test?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
api: speech Issues related to the Speech-to-Text API. samples Issues that are directly related to samples.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Test needed for speech/src/main/java/com/example/speech/InfiniteStreamRecognize.java
4 participants