blob: df549b97d2a3cfffd6f109448e53c82c2ce44cd9 [file] [log] [blame]
// Copyright 2016 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 "components/autofill/core/common/signatures_util.h"
#include <cctype>
#include "base/sha1.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/utf_string_conversions.h"
#include "components/autofill/core/common/autofill_util.h"
#include "components/autofill/core/common/form_data.h"
#include "components/autofill/core/common/form_field_data.h"
#include "url/gurl.h"
namespace autofill {
namespace {
// Returns a copy of |input| without >= 5 consecutive digits.
std::string StripDigitsIfRequired(const base::string16& input) {
std::string input_utf8 = base::UTF16ToUTF8(input);
std::string result;
result.reserve(input_utf8.length());
for (size_t i = 0; i < input_utf8.length();) {
if (std::isdigit(input_utf8[i])) {
size_t count = 0;
while (i < input_utf8.length() && std::isdigit(input_utf8[i])) {
i++;
count++;
}
if (count < 5)
result.append(input_utf8, i - count, count);
} else {
result.push_back(input_utf8[i]);
i++;
}
}
return result;
}
} // namespace
FormSignature CalculateFormSignature(const FormData& form_data) {
const GURL& target_url = form_data.action;
const GURL& source_url = form_data.origin;
std::string scheme(target_url.scheme());
std::string host(target_url.host());
// If target host or scheme is empty, set scheme and host of source url.
// This is done to match the Toolbar's behavior.
if (scheme.empty() || host.empty()) {
scheme = source_url.scheme();
host = source_url.host();
}
std::string form_signature_field_names;
for (const FormFieldData& field : form_data.fields) {
if (!ShouldSkipField(field)) {
// Add all supported form fields (including with empty names) to the
// signature. This is a requirement for Autofill servers.
form_signature_field_names.append("&");
form_signature_field_names.append(StripDigitsIfRequired(field.name));
}
}
std::string form_string = scheme + "://" + host + "&" +
base::UTF16ToUTF8(form_data.name) +
form_signature_field_names;
return StrToHash64Bit(form_string);
}
FieldSignature CalculateFieldSignatureByNameAndType(
const base::string16& field_name,
const std::string& field_type) {
std::string name = base::UTF16ToUTF8(field_name);
std::string field_string = name + "&" + field_type;
return StrToHash32Bit(field_string);
}
FieldSignature CalculateFieldSignatureForField(
const FormFieldData& field_data) {
return CalculateFieldSignatureByNameAndType(field_data.name,
field_data.form_control_type);
}
uint64_t StrToHash64Bit(const std::string& str) {
std::string hash_bin = base::SHA1HashString(str);
DCHECK_EQ(base::kSHA1Length, hash_bin.length());
uint64_t hash64 = (((static_cast<uint64_t>(hash_bin[0])) & 0xFF) << 56) |
(((static_cast<uint64_t>(hash_bin[1])) & 0xFF) << 48) |
(((static_cast<uint64_t>(hash_bin[2])) & 0xFF) << 40) |
(((static_cast<uint64_t>(hash_bin[3])) & 0xFF) << 32) |
(((static_cast<uint64_t>(hash_bin[4])) & 0xFF) << 24) |
(((static_cast<uint64_t>(hash_bin[5])) & 0xFF) << 16) |
(((static_cast<uint64_t>(hash_bin[6])) & 0xFF) << 8) |
((static_cast<uint64_t>(hash_bin[7])) & 0xFF);
return hash64;
}
uint32_t StrToHash32Bit(const std::string& str) {
std::string hash_bin = base::SHA1HashString(str);
DCHECK_EQ(base::kSHA1Length, hash_bin.length());
uint32_t hash32 = ((hash_bin[0] & 0xFF) << 24) |
((hash_bin[1] & 0xFF) << 16) | ((hash_bin[2] & 0xFF) << 8) |
(hash_bin[3] & 0xFF);
return hash32;
}
} // namespace autofill