184 lines
6.5 KiB
C++
184 lines
6.5 KiB
C++
//
|
|
// Copyright (C) 2012 The Android Open Source Project
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
#include "shill/portal_detector.h"
|
|
|
|
#include <string>
|
|
|
|
#include <base/bind.h>
|
|
#include <base/strings/string_number_conversions.h>
|
|
#include <base/strings/string_util.h>
|
|
#include <base/strings/stringprintf.h>
|
|
#if defined(__ANDROID__)
|
|
#include <dbus/service_constants.h>
|
|
#else
|
|
#include <chromeos/dbus/service_constants.h>
|
|
#endif // __ANDROID__
|
|
|
|
#include "shill/connection.h"
|
|
#include "shill/connectivity_trial.h"
|
|
#include "shill/logging.h"
|
|
|
|
using base::Bind;
|
|
using base::Callback;
|
|
using base::StringPrintf;
|
|
using std::string;
|
|
|
|
namespace shill {
|
|
|
|
namespace Logging {
|
|
static auto kModuleLogScope = ScopeLogger::kPortal;
|
|
static string ObjectID(Connection* c) { return c->interface_name(); }
|
|
}
|
|
|
|
const int PortalDetector::kDefaultCheckIntervalSeconds = 30;
|
|
const char PortalDetector::kDefaultCheckPortalList[] = "ethernet,wifi,cellular";
|
|
|
|
const int PortalDetector::kMaxRequestAttempts = 3;
|
|
const int PortalDetector::kMinTimeBetweenAttemptsSeconds = 3;
|
|
const int PortalDetector::kRequestTimeoutSeconds = 10;
|
|
const int PortalDetector::kMaxFailuresInContentPhase = 2;
|
|
|
|
PortalDetector::PortalDetector(
|
|
ConnectionRefPtr connection,
|
|
EventDispatcher* dispatcher,
|
|
const Callback<void(const PortalDetector::Result&)>& callback)
|
|
: attempt_count_(0),
|
|
attempt_start_time_((struct timeval){0}),
|
|
connection_(connection),
|
|
dispatcher_(dispatcher),
|
|
weak_ptr_factory_(this),
|
|
portal_result_callback_(callback),
|
|
connectivity_trial_callback_(Bind(&PortalDetector::CompleteAttempt,
|
|
weak_ptr_factory_.GetWeakPtr())),
|
|
time_(Time::GetInstance()),
|
|
failures_in_content_phase_(0),
|
|
connectivity_trial_(
|
|
new ConnectivityTrial(connection_,
|
|
dispatcher_,
|
|
kRequestTimeoutSeconds,
|
|
connectivity_trial_callback_)) { }
|
|
|
|
PortalDetector::~PortalDetector() {
|
|
Stop();
|
|
}
|
|
|
|
bool PortalDetector::Start(const string& url_string) {
|
|
return StartAfterDelay(url_string, 0);
|
|
}
|
|
|
|
bool PortalDetector::StartAfterDelay(const string& url_string,
|
|
int delay_seconds) {
|
|
SLOG(connection_.get(), 3) << "In " << __func__;
|
|
|
|
if (!connectivity_trial_->Start(url_string, delay_seconds * 1000)) {
|
|
return false;
|
|
}
|
|
attempt_count_ = 1;
|
|
// The attempt_start_time_ is calculated based on the current time and
|
|
// |delay_seconds|. This is used to determine if a portal detection attempt
|
|
// is in progress.
|
|
UpdateAttemptTime(delay_seconds);
|
|
// If we're starting a new set of attempts, discard past failure history.
|
|
failures_in_content_phase_ = 0;
|
|
return true;
|
|
}
|
|
|
|
void PortalDetector::Stop() {
|
|
SLOG(connection_.get(), 3) << "In " << __func__;
|
|
|
|
attempt_count_ = 0;
|
|
failures_in_content_phase_ = 0;
|
|
if (connectivity_trial_.get())
|
|
connectivity_trial_->Stop();
|
|
}
|
|
|
|
// IsInProgress returns true if a ConnectivityTrial is actively testing the
|
|
// connection. If Start has been called, but the trial was delayed,
|
|
// IsInProgress will return false. PortalDetector implements this by
|
|
// calculating the start time of the next ConnectivityTrial. After an initial
|
|
// trial and in the case where multiple attempts may be tried, IsInProgress will
|
|
// return true.
|
|
bool PortalDetector::IsInProgress() {
|
|
if (attempt_count_ > 1)
|
|
return true;
|
|
if (attempt_count_ == 1 && connectivity_trial_.get())
|
|
return connectivity_trial_->IsActive();
|
|
return false;
|
|
}
|
|
|
|
void PortalDetector::CompleteAttempt(ConnectivityTrial::Result trial_result) {
|
|
Result result = Result(trial_result);
|
|
if (trial_result.status == ConnectivityTrial::kStatusFailure &&
|
|
trial_result.phase == ConnectivityTrial::kPhaseContent) {
|
|
failures_in_content_phase_++;
|
|
}
|
|
|
|
LOG(INFO) << StringPrintf("Portal detection completed attempt %d with "
|
|
"phase==%s, status==%s, failures in content==%d",
|
|
attempt_count_,
|
|
ConnectivityTrial::PhaseToString(
|
|
trial_result.phase).c_str(),
|
|
ConnectivityTrial::StatusToString(
|
|
trial_result.status).c_str(),
|
|
failures_in_content_phase_);
|
|
|
|
if (trial_result.status == ConnectivityTrial::kStatusSuccess ||
|
|
attempt_count_ >= kMaxRequestAttempts ||
|
|
failures_in_content_phase_ >= kMaxFailuresInContentPhase) {
|
|
result.num_attempts = attempt_count_;
|
|
result.final = true;
|
|
Stop();
|
|
} else {
|
|
attempt_count_++;
|
|
int retry_delay_seconds = AdjustStartDelay(0);
|
|
connectivity_trial_->Retry(retry_delay_seconds * 1000);
|
|
UpdateAttemptTime(retry_delay_seconds);
|
|
}
|
|
portal_result_callback_.Run(result);
|
|
}
|
|
|
|
void PortalDetector::UpdateAttemptTime(int delay_seconds) {
|
|
time_->GetTimeMonotonic(&attempt_start_time_);
|
|
struct timeval delay_timeval = { delay_seconds, 0 };
|
|
timeradd(&attempt_start_time_, &delay_timeval, &attempt_start_time_);
|
|
}
|
|
|
|
|
|
int PortalDetector::AdjustStartDelay(int init_delay_seconds) {
|
|
int next_attempt_delay_seconds = 0;
|
|
if (attempt_count_ > 0) {
|
|
// Ensure that attempts are spaced at least by a minimal interval.
|
|
struct timeval now, elapsed_time;
|
|
time_->GetTimeMonotonic(&now);
|
|
timersub(&now, &attempt_start_time_, &elapsed_time);
|
|
SLOG(connection_.get(), 4) << "Elapsed time from previous attempt is "
|
|
<< elapsed_time.tv_sec << " seconds.";
|
|
if (elapsed_time.tv_sec < kMinTimeBetweenAttemptsSeconds) {
|
|
next_attempt_delay_seconds = kMinTimeBetweenAttemptsSeconds -
|
|
elapsed_time.tv_sec;
|
|
}
|
|
} else {
|
|
LOG(FATAL) << "AdjustStartDelay in PortalDetector called without "
|
|
"previous attempts";
|
|
}
|
|
SLOG(connection_.get(), 3) << "Adjusting trial start delay from "
|
|
<< init_delay_seconds << " seconds to "
|
|
<< next_attempt_delay_seconds << " seconds.";
|
|
return next_attempt_delay_seconds;
|
|
}
|
|
|
|
} // namespace shill
|