Index: components/autofill/core/browser/credit_card_field.cc |
diff --git a/components/autofill/core/browser/credit_card_field.cc b/components/autofill/core/browser/credit_card_field.cc |
index ba9c001022dc69ced0d5207c70f5d64b26d4550b..1c8d02ef8b696f9f5bd28ae4d568a206f98fd7c5 100644 |
--- a/components/autofill/core/browser/credit_card_field.cc |
+++ b/components/autofill/core/browser/credit_card_field.cc |
@@ -7,19 +7,55 @@ |
#include <stddef.h> |
#include "base/memory/scoped_ptr.h" |
+#include "base/stl_util.h" |
#include "base/strings/string16.h" |
+#include "base/strings/string_number_conversions.h" |
#include "base/strings/string_util.h" |
#include "base/strings/utf_string_conversions.h" |
+#include "base/time/time.h" |
#include "components/autofill/core/browser/autofill_field.h" |
#include "components/autofill/core/browser/autofill_regex_constants.h" |
+#include "components/autofill/core/browser/autofill_regexes.h" |
#include "components/autofill/core/browser/autofill_scanner.h" |
#include "components/autofill/core/browser/field_types.h" |
namespace autofill { |
+namespace { |
+ |
// Credit card numbers are at most 19 digits in length. |
// [Ref: http://en.wikipedia.org/wiki/Bank_card_number] |
-static const size_t kMaxValidCardNumberSize = 19; |
+const size_t kMaxValidCardNumberSize = 19; |
+ |
+// Look for the vector |regex_needles| in |haystack|. Returns true if a |
+// consecutive section of |haystack| matches |regex_needles|. |
+bool FindConsecutiveStrings(const std::vector<base::string16>& regex_needles, |
+ const std::vector<base::string16>& haystack) { |
+ if (regex_needles.empty() || haystack.empty()) |
+ return false; |
+ |
+ auto haystack_remaining_it = haystack.begin(); |
+ size_t haystack_remaining_size = haystack.size(); |
+ while (regex_needles.size() <= haystack_remaining_size) { |
Evan Stade
2015/03/18 04:29:25
for (size_t i = 0; i < haystack.size() - regex_nee
Lei Zhang
2015/03/18 18:08:56
Done.
|
+ auto haystack_it = haystack_remaining_it; |
+ bool found_match = true; |
+ for (const base::string16& regex : regex_needles) { |
+ if (!MatchesPattern(*haystack_it, regex)) { |
+ found_match = false; |
+ break; |
+ } |
+ ++haystack_it; |
+ } |
+ if (found_match) |
+ return true; |
+ |
+ ++haystack_remaining_it; |
+ --haystack_remaining_size; |
+ } |
+ return false; |
+} |
+ |
+} |
// static |
scoped_ptr<FormField> CreditCardField::Parse(AutofillScanner* scanner) { |
@@ -157,6 +193,68 @@ scoped_ptr<FormField> CreditCardField::Parse(AutofillScanner* scanner) { |
return nullptr; |
} |
+// static |
+bool CreditCardField::LikelyCardMonthSelectField(AutofillScanner* scanner) { |
+ if (scanner->IsEnd()) |
+ return false; |
+ |
+ AutofillField* field = scanner->Cursor(); |
+ if (!MatchesFormControlType(field->form_control_type, MATCH_SELECT)) |
+ return false; |
+ |
+ if (field->option_values.size() < 12 || field->option_values.size() > 13) |
+ return false; |
+ |
+ // Filter out years. |
+ const base::string16 kNumericalYearRe = |
+ base::ASCIIToUTF16("[1-9][0-9][0-9][0-9]"); |
+ for (const auto& value : field->option_values) { |
+ if (MatchesPattern(value, kNumericalYearRe)) |
+ return false; |
+ } |
+ for (const auto& value : field->option_contents) { |
+ if (MatchesPattern(value, kNumericalYearRe)) |
+ return false; |
+ } |
+ |
+ // Look for numerical months. |
+ const base::string16 kNumericalMonthRe = base::ASCIIToUTF16("12"); |
+ if (MatchesPattern(field->option_values.back(), kNumericalMonthRe) || |
+ MatchesPattern(field->option_contents.back(), kNumericalMonthRe)) { |
+ return true; |
+ } |
+ |
+ // Maybe do more matches here. e.g. look for (translated) December. |
+ |
+ // Unsure? Return false. |
+ return false; |
+} |
+ |
+// static |
+bool CreditCardField::LikelyCardYearSelectField(AutofillScanner* scanner) { |
+ if (scanner->IsEnd()) |
+ return false; |
+ |
+ AutofillField* field = scanner->Cursor(); |
+ if (!MatchesFormControlType(field->form_control_type, MATCH_SELECT)) |
+ return false; |
+ |
+ const base::Time time_now = base::Time::Now(); |
+ base::Time::Exploded time_exploded; |
+ time_now.UTCExplode(&time_exploded); |
+ |
+ const int kYearsToMatch = 3; |
+ std::vector<base::string16> years_to_check; |
+ for (int year = time_exploded.year; |
+ year < time_exploded.year + kYearsToMatch; |
+ ++year) { |
+ years_to_check.push_back(base::IntToString16(year)); |
+ } |
+ if (FindConsecutiveStrings(years_to_check, field->option_values)) |
Evan Stade
2015/03/18 04:29:25
if X
return true
return Y
is equivalent to
ret
Lei Zhang
2015/03/18 18:08:56
Done.
|
+ return true; |
+ return FindConsecutiveStrings(years_to_check, field->option_contents); |
+} |
+ |
CreditCardField::CreditCardField() |
: cardholder_(nullptr), |
cardholder_last_(nullptr), |
@@ -215,8 +313,24 @@ bool CreditCardField::ParseExpirationDate(AutofillScanner* scanner) { |
if (expiration_month_ || expiration_date_) |
return false; |
- // First try to parse split month/year expiration fields. |
+ // First try to parse split month/year expiration fields by looking for a |
+ // pair of select fields that look like month/year. |
size_t month_year_saved_cursor = scanner->SaveCursor(); |
+ |
+ if (LikelyCardMonthSelectField(scanner)) { |
+ expiration_month_ = scanner->Cursor(); |
+ scanner->Advance(); |
+ if (LikelyCardYearSelectField(scanner)) { |
+ expiration_year_ = scanner->Cursor(); |
+ scanner->Advance(); |
+ return true; |
+ } |
+ expiration_month_ = nullptr; |
+ expiration_year_ = nullptr; |
+ } |
+ |
+ // If that fails, do a general regex search. |
+ scanner->RewindTo(month_year_saved_cursor); |
const int kMatchTelAndSelect = MATCH_DEFAULT | MATCH_TELEPHONE | MATCH_SELECT; |
if (ParseFieldSpecifics(scanner, |
base::UTF8ToUTF16(kExpirationMonthRe), |