2020-12-04 07:15:23 +00:00
|
|
|
#include <folly/String.h>
|
|
|
|
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
|
|
|
#include <iterator>
|
|
|
|
#include <map>
|
|
|
|
#include <regex>
|
|
|
|
#include <set>
|
|
|
|
#include <sstream>
|
|
|
|
#include <streambuf>
|
|
|
|
#include <vector>
|
|
|
|
|
2020-12-04 07:29:07 +00:00
|
|
|
// Forward declarations of validators are needed for FIELD_VALIDATORS
|
|
|
|
// Technically this might go in a .hpp file but eh, this is a single file solution...
|
|
|
|
bool isValidBirthYear(int year);
|
|
|
|
bool isValidIssueYear(int year);
|
|
|
|
bool isValidExpirationYear(int year);
|
|
|
|
bool isValidHeight(const std::string &height);
|
|
|
|
bool isValidHairColor(const std::string &height);
|
|
|
|
bool isValidEyeColor(const std::string &height);
|
|
|
|
bool isValidPassportNumber(const std::string &height);
|
|
|
|
|
2020-12-04 07:15:23 +00:00
|
|
|
// cid is not required
|
2020-12-04 07:29:07 +00:00
|
|
|
const std::set<std::string> REQUIRED_FIELDS{"byr", "iyr", "eyr", "hgt", "hcl", "ecl", "pid"};
|
|
|
|
const std::set<std::string> VALID_EYE_COLORS{"amb", "blu", "brn", "gry", "grn", "hzl", "oth"};
|
2020-12-04 07:15:23 +00:00
|
|
|
const std::string PASSPORT_DELIM = "\n\n";
|
2020-12-04 07:29:07 +00:00
|
|
|
const std::string PASSPORT_FIELD_DELIM = ":";
|
|
|
|
const auto FIELD_VALIDATORS = std::map<std::string, std::function<bool(const std::string &)>>{
|
|
|
|
{"byr", [](const std::string &value) { return isValidBirthYear(std::stoi(value)); }},
|
|
|
|
{"iyr", [](const std::string &value) { return isValidIssueYear(std::stoi(value)); }},
|
|
|
|
{"eyr", [](const std::string &value) { return isValidExpirationYear(std::stoi(value)); }},
|
|
|
|
{"hgt", isValidHeight},
|
|
|
|
{"hcl", isValidHairColor},
|
|
|
|
{"ecl", isValidEyeColor},
|
|
|
|
{"pid", isValidPassportNumber},
|
|
|
|
{"cid", [](const std::string &value) { return true; }},
|
|
|
|
};
|
2020-12-04 07:15:23 +00:00
|
|
|
|
|
|
|
std::string read_input(const std::string &filename) {
|
|
|
|
std::vector<std::string> input;
|
|
|
|
std::string line;
|
|
|
|
std::ifstream file(filename);
|
|
|
|
return std::string(std::istreambuf_iterator<char>(file), std::istreambuf_iterator<char>());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if the given passport has all of the required fields
|
|
|
|
* @param passport The passport to check
|
|
|
|
* @return bool Whether or not he given passport has all the fields
|
|
|
|
*/
|
|
|
|
bool hasRequiredFields(const std::map<std::string, std::string> &passport) {
|
|
|
|
std::set<std::string> fields;
|
|
|
|
std::transform(
|
|
|
|
passport.begin(),
|
|
|
|
passport.end(),
|
|
|
|
std::inserter(fields, fields.end()),
|
|
|
|
[](const std::pair<const std::string, std::string> &entry) { return entry.first; });
|
|
|
|
|
|
|
|
std::set<std::string> missingFields;
|
|
|
|
std::set_difference(
|
2020-12-04 07:29:07 +00:00
|
|
|
REQUIRED_FIELDS.begin(),
|
|
|
|
REQUIRED_FIELDS.end(),
|
2020-12-04 07:15:23 +00:00
|
|
|
fields.begin(),
|
|
|
|
fields.end(),
|
|
|
|
std::inserter(missingFields, missingFields.end()));
|
|
|
|
|
|
|
|
return missingFields.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get each individual passport as a vector of strings
|
|
|
|
* @param input The passport raw from the file
|
|
|
|
* @return std::vector<std::string> All of the passports as strings
|
|
|
|
*/
|
|
|
|
std::vector<std::string> getPassportStrings(const std::string &input) {
|
|
|
|
std::vector<std::string> passports;
|
|
|
|
folly::split(PASSPORT_DELIM, input, passports);
|
|
|
|
for (std::string &passport : passports) {
|
|
|
|
boost::algorithm::trim(passport);
|
|
|
|
std::replace(passport.begin(), passport.end(), '\n', ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
return passports;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make a vector of maps of all of the passports
|
|
|
|
* @param input The raw input from the file
|
|
|
|
* @return std::vector<std::map<std::string, std::string>> All of the passports, turned into a vector of key -> value
|
|
|
|
* form
|
|
|
|
*/
|
|
|
|
std::vector<std::map<std::string, std::string>> makePassportMaps(const std::string &input) {
|
|
|
|
std::vector<std::string> passports = getPassportStrings(input);
|
|
|
|
std::vector<std::map<std::string, std::string>> passportMaps;
|
|
|
|
for (const std::string &passport : passports) {
|
|
|
|
std::vector<std::string> entries;
|
|
|
|
folly::split(" ", passport, entries);
|
|
|
|
|
2020-12-04 07:29:07 +00:00
|
|
|
// Make a map of all of the entries in this pasport
|
2020-12-04 07:15:23 +00:00
|
|
|
std::map<std::string, std::string> passportMap;
|
|
|
|
for (const std::string &entry : entries) {
|
|
|
|
std::vector<std::string> parts;
|
2020-12-04 07:29:07 +00:00
|
|
|
folly::split(PASSPORT_FIELD_DELIM, entry, parts);
|
2020-12-04 07:15:23 +00:00
|
|
|
passportMap.insert(std::pair<std::string, std::string>(parts.at(0), parts.at(1)));
|
|
|
|
}
|
|
|
|
|
|
|
|
passportMaps.push_back(passportMap);
|
|
|
|
}
|
|
|
|
|
|
|
|
return passportMaps;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidBirthYear(int year) {
|
|
|
|
return year >= 1920 && year <= 2002;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidIssueYear(int year) {
|
|
|
|
return year >= 2010 && year <= 2020;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidExpirationYear(int year) {
|
|
|
|
return year >= 2020 && year <= 2030;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidHeight(const std::string &height) {
|
|
|
|
std::string suffix = height.substr(height.length() - 2, 2);
|
|
|
|
std::string prefix = height.substr(0, height.length() - 2);
|
|
|
|
int value = std::stoi(prefix);
|
|
|
|
if (suffix == "cm") {
|
|
|
|
return value >= 150 && value <= 193;
|
|
|
|
} else if (suffix == "in") {
|
|
|
|
return value >= 59 && value <= 76;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidHairColor(const std::string &color) {
|
|
|
|
std::regex expression("#[0-9a-f]{6}");
|
|
|
|
return std::regex_match(color, expression);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidEyeColor(const std::string &color) {
|
2020-12-04 07:29:07 +00:00
|
|
|
return VALID_EYE_COLORS.find(color) != VALID_EYE_COLORS.end();
|
2020-12-04 07:15:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool isValidPassportNumber(const std::string &num) {
|
|
|
|
std::regex expression("[0-9]{9}");
|
|
|
|
return std::regex_match(num, expression);
|
|
|
|
}
|
|
|
|
|
2020-12-04 07:29:07 +00:00
|
|
|
bool isFieldValid(const std::string &fieldName, const std::string &value) {
|
|
|
|
auto validator = FIELD_VALIDATORS.find(fieldName);
|
|
|
|
if (validator == FIELD_VALIDATORS.end()) {
|
|
|
|
throw new std::invalid_argument("Bad field name");
|
2020-12-04 07:15:23 +00:00
|
|
|
}
|
|
|
|
|
2020-12-04 07:29:07 +00:00
|
|
|
return validator->second(value);
|
2020-12-04 07:15:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int part1(const std::vector<std::map<std::string, std::string>> &passports) {
|
|
|
|
int count = 0;
|
|
|
|
for (const std::map<std::string, std::string> &passport : passports) {
|
|
|
|
count += hasRequiredFields(passport);
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
int part2(const std::vector<std::map<std::string, std::string>> &passports) {
|
|
|
|
int count = 0;
|
|
|
|
for (const std::map<std::string, std::string> &passport : passports) {
|
|
|
|
if (!hasRequiredFields(passport)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool valid = true;
|
|
|
|
for (const std::pair<const std::string, std::string> entry : passport) {
|
|
|
|
if (!isFieldValid(entry.first, entry.second)) {
|
|
|
|
valid = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
count += valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
if (argc != 2) {
|
|
|
|
std::cerr << argv[0] << " <input_file>" << std::endl;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string input = read_input(argv[1]);
|
|
|
|
auto passports = makePassportMaps(input);
|
|
|
|
std::cout << part1(passports) << std::endl;
|
|
|
|
std::cout << part2(passports) << std::endl;
|
|
|
|
}
|