blob: 4dbb1e708588a962f198636192a9ee5ba3da2577 [file] [log] [blame]
// Copyright (c) 2012 The Chromium OS 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 "shill/static_ip_parameters.h"
#include <string.h>
#include <base/strings/string_split.h>
#include <base/strings/string_util.h>
#include <chromeos/dbus/service_constants.h>
#include "shill/error.h"
#include "shill/logging.h"
#include "shill/net/ip_address.h"
#include "shill/property_accessor.h"
#include "shill/property_store.h"
#include "shill/store_interface.h"
using std::string;
using std::vector;
namespace shill {
// static
const char StaticIPParameters::kConfigKeyPrefix[] = "StaticIP.";
// static
const char StaticIPParameters::kSavedConfigKeyPrefix[] = "SavedIP.";
// static
const StaticIPParameters::Property StaticIPParameters::kProperties[] = {
{ kAddressProperty, Property::kTypeString },
{ kGatewayProperty, Property::kTypeString },
{ kMtuProperty, Property::kTypeInt32 },
{ kNameServersProperty, Property::kTypeStrings },
{ kPeerAddressProperty, Property::kTypeString },
{ kPrefixlenProperty, Property::kTypeInt32 }
};
StaticIPParameters::StaticIPParameters() {}
StaticIPParameters::~StaticIPParameters() {}
void StaticIPParameters::PlumbPropertyStore(PropertyStore *store) {
// These individual fields will be deprecated once Chrome starts using
// the KeyValueStore dict directly.
for (size_t i = 0; i < arraysize(kProperties); ++i) {
const Property &property = kProperties[i];
const string name(string(kConfigKeyPrefix) + property.name);
const string saved_name(string(kSavedConfigKeyPrefix) + property.name);
switch (property.type) {
case Property::kTypeInt32:
store->RegisterDerivedInt32(
name,
Int32Accessor(
new CustomMappedAccessor<StaticIPParameters, int32_t, size_t>(
this,
&StaticIPParameters::ClearMappedProperty,
&StaticIPParameters::GetMappedInt32Property,
&StaticIPParameters::SetMappedInt32Property,
i)));
store->RegisterDerivedInt32(
saved_name,
Int32Accessor(
new CustomMappedAccessor<StaticIPParameters, int32_t, size_t>(
this,
&StaticIPParameters::ClearMappedSavedProperty,
&StaticIPParameters::GetMappedSavedInt32Property,
&StaticIPParameters::SetMappedSavedInt32Property,
i)));
break;
case Property::kTypeString:
store->RegisterDerivedString(
name,
StringAccessor(
new CustomMappedAccessor<StaticIPParameters, string, size_t>(
this,
&StaticIPParameters::ClearMappedProperty,
&StaticIPParameters::GetMappedStringProperty,
&StaticIPParameters::SetMappedStringProperty,
i)));
store->RegisterDerivedString(
saved_name,
StringAccessor(
new CustomMappedAccessor<StaticIPParameters, string, size_t>(
this,
&StaticIPParameters::ClearMappedSavedProperty,
&StaticIPParameters::GetMappedSavedStringProperty,
&StaticIPParameters::SetMappedSavedStringProperty,
i)));
break;
case Property::kTypeStrings:
// Since Chrome is still using string for the nameservers, the
// registered function will convert the string from/to string vector
// stored in the KeyValueStore.
store->RegisterDerivedString(
name,
StringAccessor(
new CustomMappedAccessor<StaticIPParameters, string, size_t>(
this,
&StaticIPParameters::ClearMappedProperty,
&StaticIPParameters::GetMappedStringsProperty,
&StaticIPParameters::SetMappedStringsProperty,
i)));
store->RegisterDerivedString(
saved_name,
StringAccessor(
new CustomMappedAccessor<StaticIPParameters, string, size_t>(
this,
&StaticIPParameters::ClearMappedSavedProperty,
&StaticIPParameters::GetMappedSavedStringsProperty,
&StaticIPParameters::SetMappedSavedStringsProperty,
i)));
break;
default:
NOTIMPLEMENTED();
break;
}
}
// Register KeyValueStore for both static ip and saved ip parameters.
store->RegisterDerivedKeyValueStore(
kSavedIPConfigProperty,
KeyValueStoreAccessor(
new CustomAccessor<StaticIPParameters, KeyValueStore>(
this, &StaticIPParameters::GetSavedIPConfig, nullptr)));
store->RegisterDerivedKeyValueStore(
kStaticIPConfigProperty,
KeyValueStoreAccessor(
new CustomAccessor<StaticIPParameters, KeyValueStore>(
this, &StaticIPParameters::GetStaticIPConfig,
&StaticIPParameters::SetStaticIPConfig)));
}
void StaticIPParameters::Load(
StoreInterface *storage, const string &storage_id) {
for (size_t i = 0; i < arraysize(kProperties); ++i) {
const Property &property = kProperties[i];
const string name(string(kConfigKeyPrefix) + property.name);
switch (property.type) {
case Property::kTypeInt32:
{
int32_t value;
if (storage->GetInt(storage_id, name, &value)) {
args_.SetInt(property.name, value);
} else {
args_.RemoveInt(property.name);
}
}
break;
case Property::kTypeString:
{
string value;
if (storage->GetString(storage_id, name, &value)) {
args_.SetString(property.name, value);
} else {
args_.RemoveString(property.name);
}
}
break;
case Property::kTypeStrings:
{
// Name servers field is stored in storage as comma separated string.
// Keep it as is to be backward compatible.
string value;
if (storage->GetString(storage_id, name, &value)) {
vector<string> string_list;
base::SplitString(value, ',', &string_list);
args_.SetStrings(property.name, string_list);
} else {
args_.RemoveStrings(property.name);
}
}
break;
default:
NOTIMPLEMENTED();
break;
}
}
}
void StaticIPParameters::Save(
StoreInterface *storage, const string &storage_id) {
for (size_t i = 0; i < arraysize(kProperties); ++i) {
const Property &property = kProperties[i];
const string name(string(kConfigKeyPrefix) + property.name);
bool property_exists = false;
switch (property.type) {
case Property::kTypeInt32:
if (args_.ContainsInt(property.name)) {
property_exists = true;
storage->SetInt(storage_id, name, args_.GetInt(property.name));
}
break;
case Property::kTypeString:
if (args_.ContainsString(property.name)) {
property_exists = true;
storage->SetString(storage_id, name, args_.GetString(property.name));
}
break;
case Property::kTypeStrings:
if (args_.ContainsStrings(property.name)) {
property_exists = true;
// Name servers field is stored in storage as comma separated string.
// Keep it as is to be backward compatible.
storage->SetString(
storage_id, name,
JoinString(args_.GetStrings(property.name), ','));
}
break;
default:
NOTIMPLEMENTED();
break;
}
if (!property_exists) {
storage->DeleteKey(storage_id, name);
}
}
}
void StaticIPParameters::ApplyInt(
const string &property, int32_t *value_out) {
saved_args_.SetInt(property, *value_out);
if (args_.ContainsInt(property)) {
*value_out = args_.GetInt(property);
}
}
void StaticIPParameters::ApplyString(
const string &property, string *value_out) {
saved_args_.SetString(property, *value_out);
if (args_.ContainsString(property)) {
*value_out = args_.GetString(property);
}
}
void StaticIPParameters::ApplyStrings(
const string &property, vector<string> *value_out) {
saved_args_.SetStrings(property, *value_out);
if (args_.ContainsStrings(property)) {
*value_out = args_.GetStrings(property);
}
}
void StaticIPParameters::ApplyTo(IPConfig::Properties *props) {
if (props->address_family == IPAddress::kFamilyUnknown) {
// In situations where no address is supplied (bad or missing DHCP config)
// supply an address family ourselves.
// TODO(pstew): Guess from the address values.
props->address_family = IPAddress::kFamilyIPv4;
}
ClearSavedParameters();
ApplyString(kAddressProperty, &props->address);
ApplyString(kGatewayProperty, &props->gateway);
ApplyInt(kMtuProperty, &props->mtu);
ApplyStrings(kNameServersProperty, &props->dns_servers);
ApplyString(kPeerAddressProperty, &props->peer_address);
ApplyInt(kPrefixlenProperty, &props->subnet_prefix);
}
void StaticIPParameters::RestoreTo(IPConfig::Properties *props) {
props->address = saved_args_.LookupString(kAddressProperty, "");
props->gateway = saved_args_.LookupString(kGatewayProperty, "");
props->mtu = saved_args_.LookupInt(kMtuProperty, 0);
props->dns_servers.clear();
if (saved_args_.ContainsStrings(kNameServersProperty)) {
props->dns_servers = saved_args_.GetStrings(kNameServersProperty);
}
props->peer_address = saved_args_.LookupString(kPeerAddressProperty, "");
props->subnet_prefix = saved_args_.LookupInt(kPrefixlenProperty, 0);
ClearSavedParameters();
}
void StaticIPParameters::ClearSavedParameters() {
saved_args_.Clear();
}
bool StaticIPParameters::ContainsAddress() const {
return args_.ContainsString(kAddressProperty) &&
args_.ContainsInt(kPrefixlenProperty);
}
bool StaticIPParameters::ContainsNameServers() const {
return args_.ContainsStrings(kNameServersProperty);
}
void StaticIPParameters::ClearMappedProperty(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const Property &property = kProperties[index];
switch (property.type) {
case Property::kTypeInt32:
if (args_.ContainsInt(property.name)) {
args_.RemoveInt(property.name);
} else {
error->Populate(Error::kNotFound, "Property is not set");
}
break;
case Property::kTypeString:
if (args_.ContainsString(property.name)) {
args_.RemoveString(property.name);
} else {
error->Populate(Error::kNotFound, "Property is not set");
}
break;
case Property::kTypeStrings:
if (args_.ContainsStrings(property.name)) {
args_.RemoveStrings(property.name);
} else {
error->Populate(Error::kNotFound, "Property is not set");
}
break;
default:
NOTIMPLEMENTED();
break;
}
}
void StaticIPParameters::ClearMappedSavedProperty(
const size_t &index, Error *error) {
error->Populate(Error::kInvalidArguments, "Property is read-only");
}
int32_t StaticIPParameters::GetMappedInt32Property(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!args_.ContainsInt(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return 0;
}
return args_.GetInt(key);
}
int32_t StaticIPParameters::GetMappedSavedInt32Property(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!saved_args_.ContainsInt(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return 0;
}
return saved_args_.GetInt(key);
}
string StaticIPParameters::GetMappedStringProperty(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!args_.ContainsString(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return string();
}
return args_.GetString(key);
}
string StaticIPParameters::GetMappedSavedStringProperty(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!saved_args_.ContainsString(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return string();
}
return saved_args_.GetString(key);
}
string StaticIPParameters::GetMappedStringsProperty(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!args_.ContainsStrings(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return string();
}
return JoinString(args_.GetStrings(key), ',');
}
string StaticIPParameters::GetMappedSavedStringsProperty(
const size_t &index, Error *error) {
CHECK(index < arraysize(kProperties));
const string &key = kProperties[index].name;
if (!saved_args_.ContainsStrings(key)) {
error->Populate(Error::kNotFound, "Property is not set");
return string();
}
return JoinString(saved_args_.GetStrings(key), ',');
}
bool StaticIPParameters::SetMappedInt32Property(
const size_t &index, const int32_t &value, Error *error) {
CHECK(index < arraysize(kProperties));
if (args_.ContainsInt(kProperties[index].name) &&
args_.GetInt(kProperties[index].name) == value) {
return false;
}
args_.SetInt(kProperties[index].name, value);
return true;
}
bool StaticIPParameters::SetMappedSavedInt32Property(
const size_t &index, const int32_t &value, Error *error) {
error->Populate(Error::kInvalidArguments, "Property is read-only");
return false;
}
bool StaticIPParameters::SetMappedStringProperty(
const size_t &index, const string &value, Error *error) {
CHECK(index < arraysize(kProperties));
if (args_.ContainsString(kProperties[index].name) &&
args_.GetString(kProperties[index].name) == value) {
return false;
}
args_.SetString(kProperties[index].name, value);
return true;
}
bool StaticIPParameters::SetMappedSavedStringProperty(
const size_t &index, const string &value, Error *error) {
error->Populate(Error::kInvalidArguments, "Property is read-only");
return false;
}
bool StaticIPParameters::SetMappedStringsProperty(
const size_t &index, const string &value, Error *error) {
CHECK(index < arraysize(kProperties));
vector<string> string_list;
base::SplitString(value, ',', &string_list);
if (args_.ContainsStrings(kProperties[index].name) &&
args_.GetStrings(kProperties[index].name) == string_list) {
return false;
}
args_.SetStrings(kProperties[index].name, string_list);
return true;
}
bool StaticIPParameters::SetMappedSavedStringsProperty(
const size_t &index, const string &value, Error *error) {
error->Populate(Error::kInvalidArguments, "Property is read-only");
return false;
}
KeyValueStore StaticIPParameters::GetSavedIPConfig(Error */*error*/) {
return saved_args_;
}
KeyValueStore StaticIPParameters::GetStaticIPConfig(Error */*error*/) {
return args_;
}
bool StaticIPParameters::SetStaticIPConfig(const KeyValueStore &value,
Error */*error*/) {
if (args_.Equals(value)) {
return false;
}
args_ = value;
return true;
}
} // namespace shill