123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- package main
- import (
- "bufio"
- "fmt"
- "log"
- "os"
- "sort"
- "strconv"
- "strings"
- )
- type rule struct {
- name string
- firstSegment [2]int
- secondSegment [2]int
- }
- var rules []rule
- func readRule(line string) {
- var newRule rule
- parts := strings.Split(line, ":")
- if len(parts) != 2 {
- log.Fatalf("Invalid line: %s", line)
- }
- newRule.name = parts[0]
- n, err := fmt.Sscanf(parts[1], "%d-%d or %d-%d\n", &newRule.firstSegment[0], &newRule.firstSegment[1], &newRule.secondSegment[0], &newRule.secondSegment[1])
- if err != nil || n != 4 {
- log.Fatalf("Error scanning '%s': %s", line, err)
- }
- rules = append(rules, newRule)
- }
- type ticket []int
- var tickets []ticket
- func readTicket(line string) {
- var newTicket ticket
- for _, item := range strings.Split(line, ",") {
- field, err := strconv.Atoi(item)
- if err != nil {
- log.Fatalf("Error parsing field from %s: %s", item, err)
- }
- newTicket = append(newTicket, field)
- }
- tickets = append(tickets, newTicket)
- }
- func readFile(file *os.File) {
- scanner := bufio.NewScanner(file)
- currentFunction := readRule
- for scanner.Scan() {
- line := scanner.Text()
- if line == "" {
- continue
- }
- if strings.Contains(line, "your ticket:") {
- currentFunction = readTicket
- continue
- }
- if strings.Contains(line, "nearby tickets:") {
- continue
- }
- currentFunction(line)
- }
- if err := scanner.Err(); err != nil {
- log.Fatalf("Scanner error: %s", err)
- }
- }
- func checkAllRulesOnField(field int) bool {
- for _, currentRule := range rules {
- if (field >= currentRule.firstSegment[0] && field <= currentRule.firstSegment[1]) || (field >= currentRule.secondSegment[0] && field <= currentRule.secondSegment[1]) {
- return true
- }
- }
- return false
- }
- var validTickets []ticket
- func sumBad() int {
- numberOfTickets := len(tickets)
- sum := 0
- for i := 1; i < numberOfTickets; i++ {
- validTicket := true
- for _, field := range tickets[i] {
- if !checkAllRulesOnField(field) {
- sum += field
- validTicket = false
- }
- }
- if validTicket {
- validTickets = append(validTickets, tickets[i])
- }
- }
- validTickets = append(validTickets, tickets[0])
- return sum
- }
- func checkRuleOnField(currentRule rule, field int) bool {
- if (field >= currentRule.firstSegment[0] && field <= currentRule.firstSegment[1]) || (field >= currentRule.secondSegment[0] && field <= currentRule.secondSegment[1]) {
- return true
- }
- return false
- }
- type fieldRules struct {
- id int
- size int
- rules []string
- }
- type bySize []fieldRules
- func (a bySize) Len() int { return len(a) }
- func (a bySize) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
- func (a bySize) Less(i, j int) bool { return a[i].size < a[j].size }
- type fieldRule struct {
- id int
- mappedRule string
- }
- var mapping []fieldRule
- func notTaken(item fieldRules) {
- for _, currentRule := range item.rules {
- new := true
- for _, takenRule := range mapping {
- if currentRule == takenRule.mappedRule {
- new = false
- break
- }
- }
- if new {
- newMapping := fieldRule{id: item.id, mappedRule: currentRule}
- mapping = append(mapping, newMapping)
- break
- }
- }
- }
- func establishOrder() {
- numberOfFields := len(rules)
- numberOfValidTickets := len(validTickets)
- var rulesByField []fieldRules
- validForField := make([]map[string]int, numberOfFields)
- for i, _ := range validForField {
- validForField[i] = make(map[string]int)
- }
- for _, item := range validTickets {
- for i := 0; i < numberOfFields; i++ {
- for _, currentRule := range rules {
- if checkRuleOnField(currentRule, item[i]) {
- validForField[i][currentRule.name]++
- }
- }
- }
- }
- for i, item := range validForField {
- current := fieldRules{id: i, size: 0}
- for key, value := range item {
- if value == numberOfValidTickets {
- current.rules = append(current.rules, key)
- current.size++
- }
- }
- rulesByField = append(rulesByField, current)
- }
- sort.Sort(bySize(rulesByField))
- for _, item := range rulesByField {
- notTaken(item)
- }
- }
- func checkMyTicket() int {
- myTicket := tickets[0]
- result := 1
- for _, currentRule := range mapping {
- if !strings.Contains(currentRule.mappedRule, "departure") {
- continue
- }
- result *= myTicket[currentRule.id]
- }
- return result
- }
- func main() {
- if len(os.Args) < 2 {
- log.Fatal("You need to specify a file!")
- }
- filePath := os.Args[1]
- file, err := os.Open(filePath)
- if err != nil {
- log.Fatalf("Failed to open %s!\n", filePath)
- }
- readFile(file)
- if err := file.Close(); err != nil {
- log.Fatalf("Failed to close file: %s", err)
- }
- fmt.Println("Part1:", sumBad())
- establishOrder()
- fmt.Println("Part2:", checkMyTicket())
- }
|