#include #include #include #include #include #include #include #include #include #include #include // 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); // cid is not required const std::set REQUIRED_FIELDS{"byr", "iyr", "eyr", "hgt", "hcl", "ecl", "pid"}; const std::set VALID_EYE_COLORS{"amb", "blu", "brn", "gry", "grn", "hzl", "oth"}; const std::string PASSPORT_DELIM = "\n\n"; const std::string PASSPORT_FIELD_DELIM = ":"; const auto FIELD_VALIDATORS = std::map>{ {"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; }}, }; std::string read_input(const std::string &filename) { std::vector input; std::string line; std::ifstream file(filename); return std::string(std::istreambuf_iterator(file), std::istreambuf_iterator()); } /** * 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 &passport) { std::set fields; std::transform( passport.begin(), passport.end(), std::inserter(fields, fields.end()), [](const std::pair &entry) { return entry.first; }); std::set missingFields; std::set_difference( REQUIRED_FIELDS.begin(), REQUIRED_FIELDS.end(), 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 All of the passports as strings */ std::vector getPassportStrings(const std::string &input) { std::vector 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> All of the passports, turned into a vector of key -> value * form */ std::vector> makePassportMaps(const std::string &input) { std::vector passports = getPassportStrings(input); std::vector> passportMaps; for (const std::string &passport : passports) { std::vector entries; folly::split(" ", passport, entries); // Make a map of all of the entries in this pasport std::map passportMap; for (const std::string &entry : entries) { std::vector parts; folly::split(PASSPORT_FIELD_DELIM, entry, parts); passportMap.insert(std::pair(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) { return VALID_EYE_COLORS.find(color) != VALID_EYE_COLORS.end(); } bool isValidPassportNumber(const std::string &num) { std::regex expression("[0-9]{9}"); return std::regex_match(num, expression); } 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"); } return validator->second(value); } int part1(const std::vector> &passports) { int count = 0; for (const std::map &passport : passports) { count += hasRequiredFields(passport); } return count; } int part2(const std::vector> &passports) { int count = 0; for (const std::map &passport : passports) { if (!hasRequiredFields(passport)) { continue; } bool valid = true; for (const std::pair 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] << " " << 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; }