blob: 83c27a254d9a6809b56cd1f6fc7a0e56eb366ef5 [file] [log] [blame]
// 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 "chrome/installer/gcapi/gcapi_omaha_experiment.h"
#include "base/check.h"
#include "base/strings/string_split.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "chrome/installer/gcapi/gcapi.h"
#include "chrome/installer/gcapi/google_update_util.h"
#include "chrome/installer/util/google_update_constants.h"
namespace {
constexpr const base::char16* kDays[] = {
STRING16_LITERAL("Sun"), STRING16_LITERAL("Mon"), STRING16_LITERAL("Tue"),
STRING16_LITERAL("Wed"), STRING16_LITERAL("Thu"), STRING16_LITERAL("Fri"),
STRING16_LITERAL("Sat")};
constexpr const base::char16* kMonths[] = {
STRING16_LITERAL("Jan"), STRING16_LITERAL("Feb"), STRING16_LITERAL("Mar"),
STRING16_LITERAL("Apr"), STRING16_LITERAL("May"), STRING16_LITERAL("Jun"),
STRING16_LITERAL("Jul"), STRING16_LITERAL("Aug"), STRING16_LITERAL("Sep"),
STRING16_LITERAL("Oct"), STRING16_LITERAL("Nov"), STRING16_LITERAL("Dec")};
// Returns the number of weeks since 2/3/2003.
int GetCurrentRlzWeek(const base::Time& current_time) {
const base::Time::Exploded february_third_2003_exploded = {2003, 2, 1, 3,
0, 0, 0, 0};
base::Time f;
bool conversion_success =
base::Time::FromUTCExploded(february_third_2003_exploded, &f);
DCHECK(conversion_success);
base::TimeDelta delta = current_time - f;
return delta.InDays() / 7;
}
bool SetExperimentLabel(const wchar_t* brand_code,
const base::string16& label,
int shell_mode) {
if (!brand_code) {
return false;
}
const bool system_level = shell_mode == GCAPI_INVOKED_UAC_ELEVATION;
base::string16 original_labels;
if (!gcapi_internals::ReadExperimentLabels(system_level, &original_labels))
return false;
// Split the original labels by the label separator.
std::vector<base::string16> entries = base::SplitString(
original_labels, base::string16(1, kExperimentLabelSeparator),
base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL);
// Keep all labels, but the one we want to add/replace.
base::string16 label_and_separator(label);
label_and_separator.push_back('=');
base::string16 new_labels;
for (const base::string16& entry : entries) {
if (!entry.empty() && !base::StartsWith(entry, label_and_separator,
base::CompareCase::SENSITIVE)) {
new_labels += entry;
new_labels += kExperimentLabelSeparator;
}
}
new_labels.append(
gcapi_internals::GetGCAPIExperimentLabel(brand_code, label));
return gcapi_internals::SetExperimentLabels(system_level, new_labels);
}
} // namespace
namespace gcapi_internals {
const wchar_t kReactivationLabel[] = L"reacbrand";
const wchar_t kRelaunchLabel[] = L"relaunchbrand";
base::string16 GetGCAPIExperimentLabel(const wchar_t* brand_code,
const base::string16& label) {
// Keeps a fixed time state for this GCAPI instance; this makes tests reliable
// when crossing time boundaries on the system clock and doesn't otherwise
// affect results of this short lived binary.
static time_t instance_time_value = 0;
if (instance_time_value == 0)
instance_time_value = base::Time::Now().ToTimeT();
base::Time instance_time = base::Time::FromTimeT(instance_time_value);
base::string16 gcapi_experiment_label;
base::SStringPrintf(&gcapi_experiment_label,
STRING16_LITERAL("%ls=%ls_%d|%ls"), label.c_str(),
brand_code, GetCurrentRlzWeek(instance_time),
BuildExperimentDateString(instance_time).c_str());
return gcapi_experiment_label;
}
} // namespace gcapi_internals
const base::char16 kExperimentLabelSeparator = ';';
bool SetReactivationExperimentLabels(const wchar_t* brand_code,
int shell_mode) {
return SetExperimentLabel(brand_code, gcapi_internals::kReactivationLabel,
shell_mode);
}
bool SetRelaunchExperimentLabels(const wchar_t* brand_code, int shell_mode) {
return SetExperimentLabel(brand_code, gcapi_internals::kRelaunchLabel,
shell_mode);
}
base::string16 BuildExperimentDateString(base::Time current_time) {
// The Google Update experiment_labels timestamp format is:
// "DAY, DD0 MON YYYY HH0:MI0:SE0 TZ"
// DAY = 3 character day of week,
// DD0 = 2 digit day of month,
// MON = 3 character month of year,
// YYYY = 4 digit year,
// HH0 = 2 digit hour,
// MI0 = 2 digit minute,
// SE0 = 2 digit second,
// TZ = 3 character timezone
base::Time::Exploded then = {};
current_time.UTCExplode(&then);
then.year += 1;
DCHECK(then.HasValidValues());
return base::StringPrintf(
STRING16_LITERAL("%s, %02d %s %d %02d:%02d:%02d GMT"),
kDays[then.day_of_week], then.day_of_month, kMonths[then.month - 1],
then.year, then.hour, then.minute, then.second);
}