blob: 479dc23ff5c1a9c21fc380591754d28730f337b5 [file] [log] [blame]
/*
* Copyright 2011 The WebRTC Project Authors. All rights reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "webrtc/libjingle/xmpp/pingtask.h"
#include "webrtc/libjingle/xmpp/constants.h"
#include "webrtc/base/logging.h"
#include "webrtc/base/scoped_ptr.h"
namespace buzz {
PingTask::PingTask(buzz::XmppTaskParentInterface* parent,
rtc::MessageQueue* message_queue,
uint32_t ping_period_millis,
uint32_t ping_timeout_millis)
: buzz::XmppTask(parent, buzz::XmppEngine::HL_SINGLE),
message_queue_(message_queue),
ping_period_millis_(ping_period_millis),
ping_timeout_millis_(ping_timeout_millis),
next_ping_time_(0),
ping_response_deadline_(0) {
ASSERT(ping_period_millis >= ping_timeout_millis);
}
bool PingTask::HandleStanza(const buzz::XmlElement* stanza) {
if (!MatchResponseIq(stanza, Jid(STR_EMPTY), task_id())) {
return false;
}
if (stanza->Attr(buzz::QN_TYPE) != buzz::STR_RESULT &&
stanza->Attr(buzz::QN_TYPE) != buzz::STR_ERROR) {
return false;
}
QueueStanza(stanza);
return true;
}
// This task runs indefinitely and remains in either the start or blocked
// states.
int PingTask::ProcessStart() {
if (ping_period_millis_ < ping_timeout_millis_) {
LOG(LS_ERROR) << "ping_period_millis should be >= ping_timeout_millis";
return STATE_ERROR;
}
const buzz::XmlElement* stanza = NextStanza();
if (stanza != NULL) {
// Received a ping response of some sort (don't care what it is).
ping_response_deadline_ = 0;
}
uint32_t now = rtc::Time();
// If the ping timed out, signal.
if (ping_response_deadline_ != 0 && now >= ping_response_deadline_) {
SignalTimeout();
return STATE_ERROR;
}
// Send a ping if it's time.
if (now >= next_ping_time_) {
rtc::scoped_ptr<buzz::XmlElement> stanza(
MakeIq(buzz::STR_GET, Jid(STR_EMPTY), task_id()));
stanza->AddElement(new buzz::XmlElement(QN_PING));
SendStanza(stanza.get());
ping_response_deadline_ = now + ping_timeout_millis_;
next_ping_time_ = now + ping_period_millis_;
// Wake ourselves up when it's time to send another ping or when the ping
// times out (so we can fire a signal).
message_queue_->PostDelayed(ping_timeout_millis_, this);
message_queue_->PostDelayed(ping_period_millis_, this);
}
return STATE_BLOCKED;
}
void PingTask::OnMessage(rtc::Message* msg) {
// Get the task manager to run this task so we can send a ping or signal or
// process a ping response.
Wake();
}
} // namespace buzz