// Copyright (c) 2012 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 "base/bind.h"
#include "base/memory/ref_counted.h"
#include "base/message_loop/message_loop.h"
#include "base/run_loop.h"
#include "base/strings/stringprintf.h"
#include "remoting/jingle_glue/iq_sender.h"
#include "remoting/jingle_glue/mock_objects.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/libjingle/source/talk/xmllite/xmlelement.h"
#include "third_party/libjingle/source/talk/xmpp/constants.h"
using ::testing::_;
using ::testing::DeleteArg;
using ::testing::InvokeWithoutArgs;
using ::testing::NotNull;
using ::testing::Return;
using ::testing::SaveArg;
using ::buzz::QName;
using ::buzz::XmlElement;
namespace remoting {
namespace {
const char kStanzaId[] = "123";
const char kNamespace[] = "chromium:testns";
const char kNamespacePrefix[] = "tes";
const char kBodyTag[] = "test";
const char kType[] = "get";
const char kTo[] = "user@domain.com";
class MockCallback {
public:
MOCK_METHOD2(OnReply, void(IqRequest* request, const XmlElement* reply));
};
MATCHER_P(XmlEq, expected, "") {
return arg->Str() == expected->Str();
}
} // namespace
class IqSenderTest : public testing::Test {
public:
IqSenderTest() {
EXPECT_CALL(signal_strategy_, AddListener(NotNull()));
sender_.reset(new IqSender(&signal_strategy_));
EXPECT_CALL(signal_strategy_, RemoveListener(
static_cast<SignalStrategy::Listener*>(sender_.get())));
}
protected:
void SendTestMessage() {
scoped_ptr<XmlElement> iq_body(
new XmlElement(QName(kNamespace, kBodyTag)));
XmlElement* sent_stanza;
EXPECT_CALL(signal_strategy_, GetNextId())
.WillOnce(Return(kStanzaId));
EXPECT_CALL(signal_strategy_, SendStanzaPtr(_))
.WillOnce(DoAll(SaveArg<0>(&sent_stanza), Return(true)));
request_ = sender_->SendIq(kType, kTo, iq_body.Pass(), base::Bind(
&MockCallback::OnReply, base::Unretained(&callback_)));
std::string expected_xml_string =
base::StringPrintf(
"<cli:iq type=\"%s\" to=\"%s\" id=\"%s\" "
"xmlns:cli=\"jabber:client\">"
"<%s:%s xmlns:%s=\"%s\"/>"
"</cli:iq>",
kType, kTo, kStanzaId, kNamespacePrefix, kBodyTag,
kNamespacePrefix, kNamespace);
EXPECT_EQ(expected_xml_string, sent_stanza->Str());
delete sent_stanza;
}
base::MessageLoop message_loop_;
MockSignalStrategy signal_strategy_;
scoped_ptr<IqSender> sender_;
MockCallback callback_;
scoped_ptr<IqRequest> request_;
};
TEST_F(IqSenderTest, SendIq) {
ASSERT_NO_FATAL_FAILURE({
SendTestMessage();
});
scoped_ptr<XmlElement> response(new XmlElement(buzz::QN_IQ));
response->AddAttr(QName(std::string(), "type"), "result");
response->AddAttr(QName(std::string(), "id"), kStanzaId);
response->AddAttr(QName(std::string(), "from"), kTo);
XmlElement* result = new XmlElement(
QName("test:namespace", "response-body"));
response->AddElement(result);
EXPECT_TRUE(sender_->OnSignalStrategyIncomingStanza(response.get()));
EXPECT_CALL(callback_, OnReply(request_.get(), XmlEq(response.get())));
base::RunLoop().RunUntilIdle();
}
TEST_F(IqSenderTest, Timeout) {
ASSERT_NO_FATAL_FAILURE({
SendTestMessage();
});
request_->SetTimeout(base::TimeDelta::FromMilliseconds(2));
EXPECT_CALL(callback_, OnReply(request_.get(), NULL))
.WillOnce(InvokeWithoutArgs(&message_loop_, &base::MessageLoop::Quit));
message_loop_.Run();
}
TEST_F(IqSenderTest, InvalidFrom) {
ASSERT_NO_FATAL_FAILURE({
SendTestMessage();
});
scoped_ptr<XmlElement> response(new XmlElement(buzz::QN_IQ));
response->AddAttr(QName(std::string(), "type"), "result");
response->AddAttr(QName(std::string(), "id"), kStanzaId);
response->AddAttr(QName(std::string(), "from"), "different_user@domain.com");
XmlElement* result = new XmlElement(
QName("test:namespace", "response-body"));
response->AddElement(result);
EXPECT_CALL(callback_, OnReply(_, _))
.Times(0);
EXPECT_FALSE(sender_->OnSignalStrategyIncomingStanza(response.get()));
base::RunLoop().RunUntilIdle();
}
} // namespace remoting