// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <list>
#include "base/bind.h"
#include "base/memory/scoped_ptr.h"
#include "base/run_loop.h"
#include "base/strings/utf_string_conversions.h"
#include "content/browser/speech/google_streaming_remote_engine.h"
#include "content/browser/speech/speech_recognition_manager_impl.h"
#include "content/browser/speech/speech_recognizer_impl.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/notification_types.h"
#include "content/public/browser/web_contents.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/content_browser_test.h"
#include "content/public/test/content_browser_test_utils.h"
#include "content/public/test/test_utils.h"
#include "content/shell/browser/shell.h"
#include "content/test/mock_google_streaming_server.h"
#include "media/audio/mock_audio_manager.h"
#include "media/audio/test_audio_input_controller_factory.h"
#include "testing/gtest/include/gtest/gtest.h"
using base::RunLoop;
namespace content {
class SpeechRecognitionBrowserTest :
public ContentBrowserTest,
public MockGoogleStreamingServer::Delegate,
public media::TestAudioInputControllerDelegate {
public:
enum StreamingServerState {
kIdle,
kTestAudioControllerOpened,
kClientConnected,
kClientAudioUpload,
kClientAudioUploadComplete,
kTestAudioControllerClosed,
kClientDisconnected
};
// MockGoogleStreamingServerDelegate methods.
virtual void OnClientConnected() OVERRIDE {
ASSERT_EQ(kTestAudioControllerOpened, streaming_server_state_);
streaming_server_state_ = kClientConnected;
}
virtual void OnClientAudioUpload() OVERRIDE {
if (streaming_server_state_ == kClientConnected)
streaming_server_state_ = kClientAudioUpload;
}
virtual void OnClientAudioUploadComplete() OVERRIDE {
ASSERT_EQ(kTestAudioControllerClosed, streaming_server_state_);
streaming_server_state_ = kClientAudioUploadComplete;
}
virtual void OnClientDisconnected() OVERRIDE {
ASSERT_EQ(kClientAudioUploadComplete, streaming_server_state_);
streaming_server_state_ = kClientDisconnected;
}
// media::TestAudioInputControllerDelegate methods.
virtual void TestAudioControllerOpened(
media::TestAudioInputController* controller) OVERRIDE {
ASSERT_EQ(kIdle, streaming_server_state_);
streaming_server_state_ = kTestAudioControllerOpened;
const int capture_packet_interval_ms =
(1000 * controller->audio_parameters().frames_per_buffer()) /
controller->audio_parameters().sample_rate();
ASSERT_EQ(GoogleStreamingRemoteEngine::kAudioPacketIntervalMs,
capture_packet_interval_ms);
FeedAudioController(500 /* ms */, /*noise=*/ false);
FeedAudioController(1000 /* ms */, /*noise=*/ true);
FeedAudioController(1000 /* ms */, /*noise=*/ false);
}
virtual void TestAudioControllerClosed(
media::TestAudioInputController* controller) OVERRIDE {
ASSERT_EQ(kClientAudioUpload, streaming_server_state_);
streaming_server_state_ = kTestAudioControllerClosed;
mock_streaming_server_->MockGoogleStreamingServer::SimulateResult(
GetGoodSpeechResult());
}
// Helper methods used by test fixtures.
GURL GetTestUrlFromFragment(const std::string fragment) {
return GURL(GetTestUrl("speech", "web_speech_recognition.html").spec() +
"#" + fragment);
}
std::string GetPageFragment() {
return shell()->web_contents()->GetURL().ref();
}
const StreamingServerState &streaming_server_state() {
return streaming_server_state_;
}
protected:
// ContentBrowserTest methods.
virtual void SetUpInProcessBrowserTestFixture() OVERRIDE {
test_audio_input_controller_factory_.set_delegate(this);
media::AudioInputController::set_factory_for_testing(
&test_audio_input_controller_factory_);
mock_streaming_server_.reset(new MockGoogleStreamingServer(this));
streaming_server_state_ = kIdle;
}
virtual void SetUpOnMainThread() OVERRIDE {
ASSERT_TRUE(SpeechRecognitionManagerImpl::GetInstance());
SpeechRecognizerImpl::SetAudioManagerForTesting(
new media::MockAudioManager(BrowserThread::GetMessageLoopProxyForThread(
BrowserThread::IO)));
}
virtual void TearDownOnMainThread() OVERRIDE {
SpeechRecognizerImpl::SetAudioManagerForTesting(NULL);
}
virtual void TearDownInProcessBrowserTestFixture() OVERRIDE {
test_audio_input_controller_factory_.set_delegate(NULL);
mock_streaming_server_.reset();
}
private:
static void FeedSingleBufferToAudioController(
scoped_refptr<media::TestAudioInputController> controller,
size_t buffer_size,
bool fill_with_noise) {
DCHECK(controller.get());
const media::AudioParameters& audio_params = controller->audio_parameters();
scoped_ptr<uint8[]> audio_buffer(new uint8[buffer_size]);
if (fill_with_noise) {
for (size_t i = 0; i < buffer_size; ++i)
audio_buffer[i] = static_cast<uint8>(127 * sin(i * 3.14F /
(16 * buffer_size)));
} else {
memset(audio_buffer.get(), 0, buffer_size);
}
scoped_ptr<media::AudioBus> audio_bus =
media::AudioBus::Create(audio_params);
audio_bus->FromInterleaved(&audio_buffer.get()[0],
audio_bus->frames(),
audio_params.bits_per_sample() / 8);
controller->event_handler()->OnData(controller.get(), audio_bus.get());
}
void FeedAudioController(int duration_ms, bool feed_with_noise) {
media::TestAudioInputController* controller =
test_audio_input_controller_factory_.controller();
ASSERT_TRUE(controller);
const media::AudioParameters& audio_params = controller->audio_parameters();
const size_t buffer_size = audio_params.GetBytesPerBuffer();
const int ms_per_buffer = audio_params.frames_per_buffer() * 1000 /
audio_params.sample_rate();
// We can only simulate durations that are integer multiples of the
// buffer size. In this regard see
// SpeechRecognitionEngine::GetDesiredAudioChunkDurationMs().
ASSERT_EQ(0, duration_ms % ms_per_buffer);
const int n_buffers = duration_ms / ms_per_buffer;
for (int i = 0; i < n_buffers; ++i) {
base::MessageLoop::current()->PostTask(FROM_HERE, base::Bind(
&FeedSingleBufferToAudioController,
scoped_refptr<media::TestAudioInputController>(controller),
buffer_size,
feed_with_noise));
}
}
SpeechRecognitionResult GetGoodSpeechResult() {
SpeechRecognitionResult result;
result.hypotheses.push_back(SpeechRecognitionHypothesis(
base::UTF8ToUTF16("Pictures of the moon"), 1.0F));
return result;
}
StreamingServerState streaming_server_state_;
scoped_ptr<MockGoogleStreamingServer> mock_streaming_server_;
media::TestAudioInputControllerFactory test_audio_input_controller_factory_;
};
// Simply loads the test page and checks if it was able to create a Speech
// Recognition object in JavaScript, to make sure the Web Speech API is enabled.
// http://crbug.com/396414
#if defined(OS_WIN) || defined(OS_MACOSX)
#define MAYBE_Precheck DISABLED_Precheck
#else
#define MAYBE_Precheck Precheck
#endif
IN_PROC_BROWSER_TEST_F(SpeechRecognitionBrowserTest, MAYBE_Precheck) {
NavigateToURLBlockUntilNavigationsComplete(
shell(), GetTestUrlFromFragment("precheck"), 2);
EXPECT_EQ(kIdle, streaming_server_state());
EXPECT_EQ("success", GetPageFragment());
}
IN_PROC_BROWSER_TEST_F(SpeechRecognitionBrowserTest, OneShotRecognition) {
NavigateToURLBlockUntilNavigationsComplete(
shell(), GetTestUrlFromFragment("oneshot"), 2);
EXPECT_EQ(kClientDisconnected, streaming_server_state());
EXPECT_EQ("goodresult1", GetPageFragment());
}
} // namespace content