diff --git a/README.md b/README.md index e0ea6f0..0b031cd 100644 --- a/README.md +++ b/README.md @@ -25,6 +25,7 @@ My solutions to [Advent of Code](https://adventofcode.com) puzzles. | 4 | [Scratchcards](https://adventofcode.com/2023/day/4) | [`day4.rs`](aoc2023/src/day4.rs) | | 5 | [If You Give A Seed ...](https://adventofcode.com/2023/day/5) | [`day5.rs`](aoc2023/src/day5.rs) | | 6 | [Wait For It](https://adventofcode.com/2023/day/6) | [`day6.rs`](aoc2023/src/day6.rs) | +| 7 | [Camel Cards](https://adventofcode.com/2023/day/7) | [`day7.rs`](aoc2023/src/day7.rs) | @@ -36,6 +37,7 @@ My solutions to [Advent of Code](https://adventofcode.com) puzzles. | 58.79 µs | 27.72 µs | | 19.61 µs | 5.510 µs | | 1.432 µs | 0.345 µs | +| 117.3 µs | 70.08 µs | @@ -47,6 +49,7 @@ My solutions to [Advent of Code](https://adventofcode.com) puzzles. | 53.39 µs | 28.44 µs | | 16.38 µs | 19.364 s | | 0.513 µs | 0.650 µs | +| 136.1 µs | 65.87 µs | diff --git a/aoc2023/Cargo.toml b/aoc2023/Cargo.toml index e6104b9..a702947 100644 --- a/aoc2023/Cargo.toml +++ b/aoc2023/Cargo.toml @@ -15,6 +15,7 @@ aoc-runner = { workspace = true } aoc-runner-derive = { workspace = true } indoc = { workspace = true } +arrayvec = { version = "0.7" } itertools = { version = "0.12" } num-integer = { version = "0.1" } rayon = { version = "1.8" } diff --git a/aoc2023/input/2023/day7.txt b/aoc2023/input/2023/day7.txt new file mode 100644 index 0000000..a39a921 --- /dev/null +++ b/aoc2023/input/2023/day7.txt @@ -0,0 +1,1000 @@ +4KTJ4 575 +38T4K 449 +4T437 860 +55954 240 +Q8K89 150 +85838 21 +68668 464 +JQT77 737 +99497 155 +A444A 653 +AA6QQ 28 +4Q39T 332 +8Q579 835 +58JQ5 397 +KJ77A 271 +88Q8T 365 +AATAJ 30 +3T582 325 +Q8AQQ 321 +KA6J4 215 +KA322 403 +54225 219 +4ATA4 225 +4222A 543 +2J9KK 749 +477AT 561 +6J699 880 +J4348 655 +K4444 175 +775AA 567 +J8AK6 334 +93A58 307 +JT43J 872 +A6662 929 +Q74JK 779 +AQAQQ 827 +55557 902 +6JAA6 512 +676T2 763 +QA3KK 181 +99939 377 +43T44 473 +AK48A 968 +44464 897 +7A454 475 +6JJQ8 496 +5KAKK 419 +77AKT 802 +K4K4K 719 +33739 959 +325K2 534 +93979 751 +QK9AK 290 +8J887 79 +8QKQK 203 +Q657Q 328 +23647 869 +4568K 686 +49374 63 +88882 578 +94A9A 680 +5T5T5 759 +95JJ5 502 +Q66QQ 771 +6A886 845 +74777 982 +2K222 49 +TQTQT 704 +J7KK2 47 +98A5J 187 +66AA6 53 +57J68 443 +Q77Q7 610 +63KJJ 661 +6K46K 803 +K6279 596 +79999 85 +K2J29 592 +3T625 980 +2Q555 236 +3TTQK 687 +924A2 762 +6KQ2K 4 +3473K 418 +22QQ2 947 +J69A7 915 +KJJKK 617 +989K5 993 +333JJ 257 +J27TT 626 +K22KK 942 +2A4A3 801 +49JJK 398 +328J2 50 +4K5Q7 491 +TT2QQ 651 +KA5JQ 773 +A277K 160 +AAQQA 873 +7775A 318 +73377 847 +2J898 366 +88Q88 958 +QQQ2Q 756 +J58AA 996 +387K3 190 +92999 149 +3Q5A9 843 +63434 865 +69992 643 +56983 619 +8T8AK 107 +25Q62 148 +444JJ 364 +24JA6 808 +J8J82 967 +JKA98 347 +455Q8 249 +T65T8 713 +763A8 500 +QAT67 936 +9TT99 143 +39TQ3 425 +2J448 831 +393T9 904 +79KQ6 628 +8JA4Q 371 +8TQ8T 829 +3K2A9 277 +98T77 970 +96T7Q 306 +QKKJK 207 +QTQTQ 791 +2J9QA 344 +T84QQ 137 +J4822 90 +77J77 800 +99994 858 +JK6J7 833 +QQ472 589 +389K6 761 +K35K4 547 +22244 284 +AT78K 164 +A848J 798 +57782 924 +AT998 660 +TQ8TT 663 +4J285 890 +Q7JQQ 960 +88JJJ 370 +JKKK7 866 +22J2J 916 +JT333 292 +555JJ 267 +7K75K 228 +4Q46J 66 +43T63 695 +T6TTT 606 +QQQ82 154 +86JKT 700 +KK464 111 +9QTJ8 541 +QJJQQ 386 +J5989 110 +4JAJT 794 +7K7K6 707 +K4484 747 +JJK9J 3 +97555 10 +6QJ74 717 +95582 975 +T7468 51 +A37A3 416 +KK888 639 +J2223 233 +8T7AA 944 +A3A78 16 +739T6 850 +945T2 654 +6J633 41 +2T222 339 +27469 851 +TT333 811 +A7JA8 129 +T6T93 995 +75488 459 +JQJ58 376 +7J6QQ 941 +TTTT8 787 +27JJJ 705 +JT3Q3 983 +7KKK6 668 +88899 162 +336K7 893 +5A35A 950 +T8T88 903 +7Q4K7 274 +22K27 429 +838J8 469 +27257 792 +J666J 560 +AA2AT 554 +22228 112 +4997K 943 +KJ3J2 431 +KQ373 984 +9J694 901 +65259 917 +58885 372 +JT634 45 +K5777 848 +33TTT 720 +33233 422 +26J7K 114 +K8T45 170 +92A5K 480 +84884 218 +94739 739 +3472J 815 +J5AT4 383 +8QJ3Q 946 +T99T6 891 +46674 135 +T66J7 125 +738QQ 100 +TT4T8 490 +98268 765 +465T6 477 +J77QQ 624 +9K98J 199 +J6666 176 +23KTJ 69 +7779Q 563 +46T7J 871 +8QQQQ 442 +84TK8 965 +KA3T9 326 +AKT25 448 +2J65J 253 +7KK77 714 +74Q44 36 +A498A 684 +359AK 615 +TTTQ5 874 +J22JJ 121 +25247 59 +AJ322 613 +26222 27 +A57T5 355 +89886 544 +K9333 104 +36J98 566 +KQKJ4 142 +22276 716 +Q7222 436 +AAAA4 499 +66KK3 68 +89629 909 +T9KK9 208 +5ATT8 646 +69J59 894 +J68T6 744 +57A2K 454 +93585 92 +JQT63 485 +4K4AA 877 +9936K 986 +6KK6K 65 +44484 841 +76557 672 +8T273 882 +QQ9QQ 956 +K6549 863 +3QJ39 324 +KKKJK 565 +737Q7 562 +99494 641 +TJTJT 445 +46AT4 728 +8K2AT 896 +2QT22 590 +K7JTK 156 +2K929 503 +TT2JT 799 +JAK7A 213 +32K59 470 +87899 881 +K2KK4 579 +K3AJ5 588 +68666 461 +Q4QQK 727 +QA9K8 261 +95999 122 +5A446 885 +3J335 291 +8K685 453 +K5KA5 231 +37KJK 586 +KQ6Q6 532 +QKKKA 82 +9KJ36 846 +83JJ6 202 +33T33 466 +QQQAQ 439 +Q4A94 282 +35AJT 577 +47ATQ 753 +44AAQ 745 +3AA99 330 +88533 774 +T9T6T 971 +5KT6T 685 +A999Q 913 +59555 44 +6J66Q 196 +TA542 268 +55598 487 +926A7 523 +4Q3J9 693 +47274 582 +2Q999 830 +J599T 574 +T6TTJ 116 +6K5A2 358 +88J8A 580 +T22K2 161 +444A4 884 +A5J7A 842 +3JT3T 101 +TATTA 279 +82A82 531 +A9JTA 120 +KK4KT 60 +88J8K 314 +JKKJJ 722 +37797 467 +AA999 555 +52543 145 +55Q55 83 +4444J 408 +TTT97 232 +TKTTK 243 +8J882 602 +AJK6Q 724 +32TK9 536 +7569Q 823 +37777 410 +27772 462 +6T6JA 932 +99448 797 +69669 659 +83T89 382 +QQ68Q 392 +Q94TK 390 +4TTTT 8 +Q7QA7 998 +8JT8A 237 +K7J77 74 +AT5TT 518 +6KK66 336 +83388 89 +867Q2 629 +Q9889 409 +8J7A6 616 +333Q3 511 +333KQ 879 +9TAT9 319 +KQJ46 819 +2AA4A 93 +27774 849 +2T6J4 194 +TTKKA 597 +K9437 535 +K4J48 48 +A5J52 86 +22266 952 +AQ6Q6 245 +QJ24T 669 +QA779 638 +Q4263 694 +QQ7QQ 421 +9942J 174 +TJTJK 173 +T96A9 832 +55553 608 +K78KA 127 +444QQ 305 +AA99A 516 +33J22 348 +Q7TQT 25 +87888 793 +86884 217 +94Q99 742 +A3336 514 +K6492 748 +36363 201 +JT657 474 +AK766 905 +Q333J 788 +494Q4 171 +67776 144 +33939 701 +J8265 557 +97KK7 197 +4T5TJ 614 +44494 109 +99J79 192 +3A9Q9 977 +77783 205 +A4575 14 +8K9QT 226 +8555J 32 +JAAAA 818 +4T4T4 428 +6J6J2 378 +55534 471 +76J5J 752 +3J58J 189 +7A447 681 +9T5TJ 501 +43KAJ 222 +TTJT5 524 +QA9Q5 755 +93Q74 23 +34344 781 +AQT3A 696 +A5855 766 +T532J 259 +4J278 839 +4J56Q 645 +5J588 853 +67K3A 238 +88938 674 +J8A3A 852 +2883T 402 +8Q337 637 +TTT9T 732 +T3666 978 +537J8 128 +56794 167 +3AAQQ 356 +38J38 593 +JJAAA 153 +7676A 301 +AT79K 906 +T6T3T 962 +K845K 95 +4KT79 573 +T5956 825 +2J424 969 +A7847 478 +8T8K8 919 +T88T4 310 +J92J2 368 +6Q636 64 +5943Q 56 +3KKK4 786 +45542 746 +Q6838 551 +8665J 1000 +A3757 312 +KAT4Q 721 +2T623 488 +887J7 252 +5JK55 495 +389J5 353 +77877 13 +8A2QJ 103 +JQ798 342 +85TQ2 709 +QKJ63 39 +5Q5Q5 836 +66762 940 +33AAA 293 +TJTTT 472 +6A7Q2 520 +54455 671 +334QA 75 +JQ455 169 +A4447 360 +77755 484 +KK8K8 394 +2KAAA 289 +7KJ44 886 +43444 286 +5T862 569 +333K3 113 +A5J85 508 +4A3J2 105 +42J92 224 +95922 255 +5947A 961 +K3JJ7 507 +44J49 581 +4474T 712 +AT7TT 273 +74667 925 +JT737 990 +6T222 991 +TT2JJ 70 +52225 159 +66J6A 743 +53533 227 +9J99Q 600 +85888 840 +22527 287 +67A52 895 +8888A 406 +Q5Q8Q 652 +J77JJ 807 +K2252 605 +AA5AA 381 +66966 505 +T87A5 933 +K2848 620 +JQQTQ 140 +JT889 953 +7948K 989 +8648J 393 +AAQAA 862 +69969 42 +9Q998 251 +45KKK 157 +362QA 907 +JQ9KQ 430 +99JQA 838 +QKKQ4 195 +86688 463 +5736T 126 +J5929 62 +6TAQK 198 +378K4 537 +TA3T3 84 +3QQQ8 733 +83T32 178 +K3K23 997 +TAAAA 618 +3J6J3 979 +89QQ9 810 +7Q755 299 +7A875 423 +93275 813 +89K8A 118 +KKTKT 320 +Q6K68 689 +9TK58 481 +47927 612 +2A3AJ 515 +3K343 585 +52554 859 +A7A77 265 +Q88AT 938 +4T3JQ 601 +Q9333 350 +A2A46 374 +Q9JT9 327 +KQ286 254 +QQ2J8 837 +3J333 784 +7777K 587 +252JQ 527 +42222 985 +87QQ8 939 +6JK35 482 +5KTQ9 139 +QQ4JQ 647 +93J37 546 +KTA8J 782 +5A597 741 +A4JAA 385 +QTAQT 29 +AAAKA 545 +7722Q 132 +QQ44J 401 +4Q3QQ 43 +J5494 244 +3K884 844 +24662 828 +2K552 591 +59K24 108 +4T43T 816 +AAJ44 434 +KQKQQ 285 +33433 340 +33449 414 +9TTTJ 568 +JTT64 177 +45J54 730 +66965 492 +TJJK2 313 +A8KQ2 298 +JQ683 564 +927KK 777 +JJJJJ 550 +888J6 679 +5KJ7K 834 +QQQQ6 650 +KKK6K 625 +9KJKA 678 +47Q77 35 +6262J 163 +47T3A 78 +84J82 166 +594KA 649 +27272 607 +9JTT2 420 +9K9KK 180 +6238A 22 +893KJ 987 +4544J 389 +88KT6 783 +696AJ 611 +7A2A7 486 +7845J 335 +T3488 927 +5AAA5 52 +T6TT5 234 +JTT2A 296 +7J747 456 +28J22 785 +Q8AQ9 908 +72JJT 898 +A86J5 440 +94545 94 +T4553 71 +2J96Q 216 +95T45 264 +KKJ2K 223 +JA59A 682 +722AA 692 +3434T 572 +8KK97 806 +2TA2J 517 +3KKTK 526 +ATJ6K 19 +A9K9A 124 +4K943 235 +9A999 395 +74K7K 530 +666J2 757 +9J888 627 +5J572 734 +J8888 387 +478QA 417 +82K5A 667 +KK47K 809 +J3743 81 +T2TA8 945 +A557A 698 +93939 583 +488KK 384 +46464 778 +76665 427 +2J4A4 367 +89663 632 +24444 450 +9K8T6 444 +3393T 438 +545J5 910 +QQJQQ 821 +5A4J4 972 +AAQJA 379 +79977 40 +AT33A 812 +6J669 856 +6JKK7 211 +J2K22 61 +28228 930 +J69AA 210 +7348J 657 +975J5 37 +6J787 920 +7J7J7 34 +96J2J 458 +8888K 117 +4Q867 375 +JA2AT 58 +38535 294 +Q5TT7 17 +AK58K 9 +45KQ2 57 +4J477 411 +29TA7 750 +88Q2T 644 +Q39Q6 767 +56T87 258 +37997 715 +77T77 814 +63KJT 489 +22722 963 +K9999 723 +57J55 887 +79J94 12 +Q77AA 278 +7ATQ3 861 +Q5QQ4 570 +AAA7A 457 +84267 91 +5555J 146 +45544 246 +3558A 106 +J97A2 179 +T3422 2 +78878 441 +44A4Q 796 +QJQ8Q 519 +KQ83Q 333 +3K4K3 283 +A4AA4 548 +72K63 864 +854T7 622 +68968 373 +8K7J2 18 +K923Q 186 +22229 131 +AJ7KT 621 +K8KJ7 559 +4Q444 540 +TT6AK 921 +55J25 804 +66998 900 +555KK 736 +J33A3 388 +J3677 6 +K85A7 665 +K94TJ 735 +T6AQT 415 +5Q68Q 133 +T38T9 256 +349T7 677 +9QTQ2 99 +63644 413 +68J86 876 +63366 1 +AKAJT 703 +25555 248 +Q4788 239 +TTTKT 214 +TAT3K 957 +29292 315 +QATQQ 329 +5ATA5 15 +3A3A2 183 +66787 424 +5QT9T 775 +55K35 934 +K5T4Q 230 +667Q6 115 +Q83J8 275 +6555J 522 +44AK4 711 +TT2TT 923 +JA33A 432 +2272J 922 +58787 688 +423QA 528 +7JK55 529 +68T9J 297 +89777 280 +655T5 483 +33384 948 +46258 404 +TT77T 955 +8AAAA 918 +JQ478 928 +62422 38 +2Q2QQ 754 +6QT54 630 +Q4Q4Q 191 +3J22K 826 +4J499 889 +26J25 875 +69T2K 184 +93333 702 +425TK 726 +Q745J 220 +85J29 346 +T4858 935 +43A38 634 +7JA7A 165 +JQ6QQ 247 +T6282 46 +J9343 363 +84383 300 +7J737 609 +Q9A89 123 +T8T8T 77 +95AK7 98 +87A86 345 +5482A 304 +33338 542 +9TJQ6 188 +9883J 26 +5J537 914 +9JQ45 820 +T2T2A 316 +2KKKK 209 +7529K 974 +7677Q 263 +88QK8 912 +486Q6 764 +AA3TJ 331 +47334 7 +JK8J8 455 +KKQJ5 740 +99J9T 999 +J8AKA 805 +Q5TQJ 966 +6T66T 988 +74J39 892 +A7JA6 357 +534QQ 433 +3J586 994 +TJ32Q 193 +K3KKK 738 +J8488 55 +93AA6 399 +28J82 595 +A7328 262 +366J6 656 +KJ397 640 +37A77 72 +696AA 412 +76J77 426 +K3444 676 +QJQKQ 168 +AA7KK 633 +KKJAA 951 +32Q2Q 31 +5Q6K9 981 +58252 636 +8A887 11 +67788 479 +KAQKA 760 +K7KK7 635 +K2555 666 +5K226 683 +27JT2 521 +AKA88 308 +938Q4 776 +JA5A5 790 +AK66K 468 +63JT5 772 +9Q47K 33 +5K2JQ 405 +36K63 354 +7T8TT 270 +68644 212 +2K2K7 460 +A7ATT 172 +J8K8T 949 +62656 780 +444JT 276 +K8KK7 690 +48K3Q 822 +AQQ86 525 +6A98J 899 +22256 789 +32794 673 +2835K 20 +45456 362 +88555 200 +Q635T 396 +QT646 662 +KK8KK 556 +QQ282 76 +655J6 119 +33363 931 +AA8J8 599 +A2A22 182 +2A293 623 +T849J 96 +52286 699 +82A59 138 +QTQQ8 369 +Q844T 795 +6Q666 67 +KQQQQ 549 +8TA8A 992 +74444 338 +J6543 498 +43JTT 80 +85555 710 +Q6QKQ 134 +QQ3Q3 691 +Q3499 576 +6TQ48 706 +QQQ37 758 +J888J 642 +8Q88Q 494 +26293 708 +J66J3 343 +QT464 317 +TTT59 558 +QA9KQ 309 +3TJTT 883 +9655K 204 +J6999 311 +KTT9K 584 +K9K92 868 +97752 855 +AAT6T 504 +9298A 976 +T37J3 337 +467K7 141 +99QQ9 250 +KJ29Q 857 +22J42 295 +4T27A 870 +2J533 888 +TAA98 631 +K3336 768 +544A5 926 +42657 725 +8533K 260 +AAA2J 400 +57995 452 +4JKJQ 509 +QQK59 867 +TQ469 206 +A844A 380 +263TJ 731 +79J27 185 +9J562 269 +2J222 552 +J7494 447 +TAA3T 493 +2A49Q 658 +33299 435 +5QQQQ 513 +9999J 281 +T5KK2 341 +KJJJ6 451 +K7K5Q 323 +58QKA 648 +JK5KK 24 +K337K 937 +QJ468 302 +KA43T 854 +3TTTT 697 +9KKKK 5 +9T3TT 352 +48788 152 +KKJQA 361 +9954J 349 +TTQJ5 538 +QTTTT 229 +6Q976 533 +4666J 221 +A3323 351 +Q7778 770 +55JTK 73 +5K593 594 +222A2 878 +3KQJK 598 +7K53K 446 +AAJA8 817 +3Q269 272 +QKQAQ 729 +QQ9Q5 675 +7J7A7 147 +JJ99K 603 +58T8T 303 +33572 911 +94J99 824 +9JA33 130 +J4446 670 +J5A9Q 158 +6AAJA 359 +K48KK 954 +QJQ79 102 +7KKAK 964 +56Q73 242 +AJKJK 322 +6TT84 539 +757J7 437 +6T666 718 +TA2QA 476 +23327 241 +44225 510 +949A9 571 +399Q9 97 +J97J9 391 +9AA5A 664 +89599 288 +3JA4T 87 +T3869 151 +9A599 88 +8K884 465 +KTKKJ 973 +78K65 506 +AAAA9 407 +22322 553 +KQ752 54 +KKK55 136 +8K76K 604 +555TK 266 +49K3A 497 +8JJJJ 769 diff --git a/aoc2023/src/day7.rs b/aoc2023/src/day7.rs new file mode 100644 index 0000000..90c6fb6 --- /dev/null +++ b/aoc2023/src/day7.rs @@ -0,0 +1,311 @@ +use aoc_runner_derive::{aoc, aoc_generator}; + +#[aoc_generator(day7, part1)] +fn parse1(input: &str) -> Vec> { + utils::parse(input) +} + +#[aoc(day7, part1)] +fn part1(input: &[utils::CardHandWithBid]) -> u64 { + utils::solve(input) +} + +#[aoc_generator(day7, part2)] +fn parse2(input: &str) -> Vec> { + utils::parse(input) +} + +#[aoc(day7, part2)] +fn part2(input: &[utils::CardHandWithBid]) -> u64 { + utils::solve(input) +} + +mod utils { + use itertools::Itertools; + use strum::{EnumIter, IntoEnumIterator}; + + pub fn parse>(input: &str) -> Vec> + where + HandType: From<[T; 5]>, + { + input + .lines() + .map(|line| { + let (cards, bid) = line.split_ascii_whitespace().collect_tuple().unwrap(); + let cards: [T; 5] = cards + .chars() + .map(|card| { + card.try_into() + .unwrap_or_else(|_| panic!("Invalid card: {card}")) + }) + .collect::>() + .try_into() + .unwrap_or_else(|_| panic!("Invalid hand: {cards}")); + let hand_type = cards.into(); + CardHandWithBid { + hand: CardHand { hand_type, cards }, + bid: bid.parse().unwrap(), + } + }) + .collect() + } + + pub fn solve(input: &[CardHandWithBid]) -> u64 { + input + .iter() + .sorted_by_key(|card_hand| card_hand.hand) + .enumerate() + .map(|(i, card_hand)| u64::try_from(i + 1).unwrap() * card_hand.bid) + .sum() + } + + pub struct CardHandWithBid { + pub hand: CardHand, + pub bid: u64, + } + + #[derive(Clone, Copy, Eq)] + pub struct CardHand { + pub hand_type: HandType, + pub cards: [T; 5], + } + + #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)] + pub enum HandType { + HighCard, + OnePair, + TwoPair, + ThreeOfKind, + FullHouse, + FourOfKind, + FiveOfKind, + } + + impl PartialEq for CardHand { + fn eq(&self, other: &Self) -> bool { + self.cards.eq(&other.cards) + } + } + + impl PartialOrd for CardHand { + fn partial_cmp(&self, other: &Self) -> Option { + Some(self.cmp(other)) + } + } + + impl Ord for CardHand { + fn cmp(&self, other: &Self) -> std::cmp::Ordering { + match self.hand_type.cmp(&other.hand_type) { + std::cmp::Ordering::Equal => { + for i in 0..5 { + match self.cards[i].cmp(&other.cards[i]) { + std::cmp::Ordering::Equal => {} + other => return other, + } + } + std::cmp::Ordering::Equal + } + other => other, + } + } + } + + pub mod part1 { + use super::{EnumIter, HandType, IntoEnumIterator}; + + #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, EnumIter)] + pub enum CamelCard { + Two, + Three, + Four, + Five, + Six, + Seven, + Eight, + Nine, + T, + J, + Q, + K, + A, + } + + impl TryFrom for CamelCard { + type Error = &'static str; + fn try_from(value: char) -> Result { + match value { + '2' => Ok(Self::Two), + '3' => Ok(Self::Three), + '4' => Ok(Self::Four), + '5' => Ok(Self::Five), + '6' => Ok(Self::Six), + '7' => Ok(Self::Seven), + '8' => Ok(Self::Eight), + '9' => Ok(Self::Nine), + 'T' => Ok(Self::T), + 'J' => Ok(Self::J), + 'Q' => Ok(Self::Q), + 'K' => Ok(Self::K), + 'A' => Ok(Self::A), + _ => Err("Invalid card"), + } + } + } + + impl From<[CamelCard; 5]> for HandType { + fn from(value: [CamelCard; 5]) -> Self { + let mut hand_types_raw = arrayvec::ArrayVec::::new(); + for card in CamelCard::iter() { + match value.iter().filter(|&&c| c == card).count() { + 2 => hand_types_raw.push(Self::OnePair), + 3 => hand_types_raw.push(Self::ThreeOfKind), + 4 => { + hand_types_raw.push(Self::FourOfKind); + break; + } + 5 => { + hand_types_raw.push(Self::FiveOfKind); + break; + } + _ => {} + } + } + match hand_types_raw.len() { + 0 => Self::HighCard, + 1 => hand_types_raw[0], + 2 => { + hand_types_raw.sort(); + match hand_types_raw[0] { + Self::OnePair => match hand_types_raw[1] { + Self::OnePair => Self::TwoPair, + Self::ThreeOfKind => Self::FullHouse, + _ => unreachable!(), + }, + _ => unreachable!(), + } + } + _ => unreachable!(), + } + } + } + } + + pub mod part2 { + use super::{EnumIter, HandType, IntoEnumIterator}; + + #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, EnumIter)] + pub enum CamelCard { + J, + Two, + Three, + Four, + Five, + Six, + Seven, + Eight, + Nine, + T, + Q, + K, + A, + } + + impl TryFrom for CamelCard { + type Error = &'static str; + fn try_from(value: char) -> Result { + match value { + 'J' => Ok(Self::J), + '2' => Ok(Self::Two), + '3' => Ok(Self::Three), + '4' => Ok(Self::Four), + '5' => Ok(Self::Five), + '6' => Ok(Self::Six), + '7' => Ok(Self::Seven), + '8' => Ok(Self::Eight), + '9' => Ok(Self::Nine), + 'T' => Ok(Self::T), + 'Q' => Ok(Self::Q), + 'K' => Ok(Self::K), + 'A' => Ok(Self::A), + _ => Err("Invalid card"), + } + } + } + + impl From<[CamelCard; 5]> for HandType { + fn from(value: [CamelCard; 5]) -> Self { + let mut hand_types_raw = arrayvec::ArrayVec::::new(); + let mut n_jokers = 0; + for card in CamelCard::iter() { + let n_cards = value.iter().filter(|&&c| c == card).count(); + if card == CamelCard::J { + n_jokers = n_cards; + } + match n_cards { + 2 => hand_types_raw.push(Self::OnePair), + 3 => hand_types_raw.push(Self::ThreeOfKind), + 4 => { + hand_types_raw.push(Self::FourOfKind); + break; + } + 5 => { + hand_types_raw.push(Self::FiveOfKind); + break; + } + _ => {} + } + } + let hand_type = match hand_types_raw.len() { + 0 => Self::HighCard, + 1 => hand_types_raw[0], + 2 => { + hand_types_raw.sort(); + match hand_types_raw[0] { + Self::OnePair => match hand_types_raw[1] { + Self::OnePair => Self::TwoPair, + Self::ThreeOfKind => Self::FullHouse, + _ => unreachable!(), + }, + _ => unreachable!(), + } + } + _ => unreachable!(), + }; + match (hand_type, n_jokers) { + (Self::FourOfKind, 1 | 4) | (Self::FullHouse, 2 | 3) => Self::FiveOfKind, + (Self::FullHouse, 1) | (Self::ThreeOfKind, 1 | 3) | (Self::TwoPair, 2) => { + Self::FourOfKind + } + (Self::TwoPair, 1) => Self::FullHouse, + (Self::OnePair, 1 | 2) => Self::ThreeOfKind, + (Self::HighCard, 1) => Self::OnePair, + (hand, _) => hand, + } + } + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use indoc::indoc; + + const SAMPLE: &str = indoc! {" + 32T3K 765 + T55J5 684 + KK677 28 + KTJJT 220 + QQQJA 483 + "}; + + #[test] + fn part1_example() { + assert_eq!(part1(&parse1(SAMPLE)), 6440); + } + + #[test] + fn part2_example() { + assert_eq!(part2(&parse2(SAMPLE)), 5905); + } +} diff --git a/aoc2023/src/lib.rs b/aoc2023/src/lib.rs index 5d65c09..87a0d0d 100644 --- a/aoc2023/src/lib.rs +++ b/aoc2023/src/lib.rs @@ -4,5 +4,6 @@ pub mod day3; pub mod day4; pub mod day5; pub mod day6; +pub mod day7; aoc_runner_derive::aoc_lib! { year = 2023 }