Skip to content

Commit

Permalink
Day25 (#21)
Browse files Browse the repository at this point in the history
* Add puzzle1

* Create algorithm

* Compute position on paper.

* Write getNextCode

* Implement getCodeOnInputPosition

* Solve simple input

* Solve puzzle1
  • Loading branch information
lukasbicus authored Oct 14, 2024
1 parent ea6b52f commit 53bb66f
Show file tree
Hide file tree
Showing 3 changed files with 202 additions and 0 deletions.
77 changes: 77 additions & 0 deletions scripts/aoc2015/day25/puzzle1.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
/*
--- Day 25: Let It Snow ---
Merry Christmas! Santa is booting up his weather machine; looks like you might get a white Christmas after all.
The weather machine beeps! On the console of the machine is a copy protection message asking you to enter a code from the instruction manual. Apparently, it refuses to run unless you give it that code. No problem; you'll just look up the code in the--
"Ho ho ho", Santa ponders aloud. "I can't seem to find the manual."
You look up the support number for the manufacturer and give them a call. Good thing, too - that 49th star wasn't going to earn itself.
"Oh, that machine is quite old!", they tell you. "That model went out of support six minutes ago, and we just finished shredding all of the manuals. I bet we can find you the code generation algorithm, though."
After putting you on hold for twenty minutes (your call is very important to them, it reminded you repeatedly), they finally find an engineer that remembers how the code system works.
The codes are printed on an infinite sheet of paper, starting in the top-left corner. The codes are filled in by diagonals: starting with the first row with an empty first box, the codes are filled in diagonally up and to the right. This process repeats until the infinite paper is covered. So, the first few codes are filled in in this order:
| 1 2 3 4 5 6
---+---+---+---+---+---+---+
1 | 1 3 6 10 15 21
2 | 2 5 9 14 20
3 | 4 8 13 19
4 | 7 12 18
5 | 11 17
6 | 16
For example, the 12th code would be written to row 4, column 2; the 15th code would be written to row 1, column 5.
The voice on the other end of the phone continues with how the codes are actually generated. The first code is 20151125. After that, each code is generated by taking the previous one, multiplying it by 252533, and then keeping the remainder from dividing that value by 33554393.
So, to find the second code (which ends up in row 2, column 1), start with the previous value, 20151125. Multiply it by 252533 to get 5088824049625. Then, divide that by 33554393, which leaves a remainder of 31916031. That remainder is the second code.
"Oh!", says the voice. "It looks like we missed a scrap from one of the manuals. Let me read it to you." You write down his numbers:
| 1 2 3 4 5 6
---+---------+---------+---------+---------+---------+---------+
1 | 20151125 18749137 17289845 30943339 10071777 33511524
2 | 31916031 21629792 16929656 7726640 15514188 4041754
3 | 16080970 8057251 1601130 7981243 11661866 16474243
4 | 24592653 32451966 21345942 9380097 10600672 31527494
5 | 77061 17552253 28094349 6899651 9250759 31663883
6 | 33071741 6796745 25397450 24659492 1534922 27995004
"Now remember", the voice continues, "that's not even all of the first few numbers; for example, you're missing the one at 7,1 that would come before 6,2. But, it should be enough to let your-- oh, it's time for lunch! Bye!" The call disconnects.
Santa looks nervous. Your puzzle input contains the message on the machine's console. What code do you give the machine?
*/

// Algorithm

// create a structure, that will hold results -> do we need it? no

// create a function, that will provide you next position on paper ->
// start with [0, 0]

// y/x
// y0, x0, x1, x3, ...
// y1
// y2
// y3
// ...

// continue, computing codes till you get to required position (from puzzle input)

// to get next code, you will need prev code and use formula:
// newCode = (prevCode * 252533) % 33554393

import { getCodeOnInputPosition, Position } from "./utils.ts";

const simpleInput: Position = { x: 0, y: 1 };

const puzzle1Input: Position = { x: 3082, y: 2977 };

const simpleResult = getCodeOnInputPosition(simpleInput, 20151125);
console.log(`simpleResult ${simpleResult}`);

const puzzle1Result = getCodeOnInputPosition(puzzle1Input, 20151125);
console.log(`puzzle1Result ${puzzle1Result}`);
56 changes: 56 additions & 0 deletions scripts/aoc2015/day25/utils.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
import { assertEquals } from "@std/assert";
import { describe, it } from "@std/testing/bdd";
import { getCodeOnInputPosition, getNextCode, getPosition } from "./utils.ts";

describe("getPosition", function () {
it("should return positions till { x: 1, y: 1 }", function () {
const expectedPositions = [
{ x: 0, y: 0 },
{ x: 0, y: 1 },
{ x: 1, y: 0 },
{ x: 0, y: 2 },
{ x: 1, y: 1 },
];

assertEquals([...getPosition({ x: 1, y: 1 })], expectedPositions);
});
it("should return positions till { x: 2, y: 3 }", function () {
const expectedPositions = [
{ x: 0, y: 0 },
{ x: 0, y: 1 },
{ x: 1, y: 0 },
{ x: 0, y: 2 },
{ x: 1, y: 1 },
{ x: 2, y: 0 },
{ x: 0, y: 3 },
{ x: 1, y: 2 },
{ x: 2, y: 1 },
{ x: 3, y: 0 },
{ x: 0, y: 4 },
{ x: 1, y: 3 },
{ x: 2, y: 2 },
{ x: 3, y: 1 },
{ x: 4, y: 0 },
{ x: 0, y: 5 },
{ x: 1, y: 4 },
{ x: 2, y: 3 },
];

assertEquals([...getPosition({ x: 2, y: 3 })], expectedPositions);
});
});

describe("getNextCode", function () {
it("should return valid code", function () {
assertEquals(getNextCode(20151125), 31916031);
});
});

describe("getCodeOnInputPosition", function () {
it("Should return code on position {x: 0, y: 0}", function () {
assertEquals(getCodeOnInputPosition({ x: 0, y: 0 }, 20151125), 20151125);
});
it("Should return code on position {x: 2, y: 3}", function () {
assertEquals(getCodeOnInputPosition({ x: 2, y: 3 }, 20151125), 21345942);
});
});
69 changes: 69 additions & 0 deletions scripts/aoc2015/day25/utils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
export type Position = {
x: number;
y: number;
};

// y/x
// y0, x0, x1, x3, ...
// y1
// y2
// y3
// ...

// counter is 0; i is 0
// {0, 0}

// counter is 1; i loops 0..1
// {1, 0}
// {0, 1}

// counter is 2; i loops 0..2
// {2, 0}
// {1, 1}
// {0, 2}

export function* getPosition(finalPosition: Position): Generator<Position> {
let counter = 0;
let position: Position;
do {
for (let i = 0; i <= counter; ++i) {
position = { x: i, y: counter - i };
yield position;
if (position.x === finalPosition.x && position.y === finalPosition.y) {
return;
}
}
counter++;
} while (
true
);
}

export function getNextCode(prevCode: number): number {
return (prevCode * 252533) % 33554393;
}

export function getCodeOnInputPosition(
finalPosition: Position,
initialCode: number,
): number {
if (finalPosition.x === 0 && finalPosition.y === 0) {
// if final position is 0, 0 return initialCode
return initialCode;
}
// loop positions till final position
let prevCode: number = initialCode;
for (const position of getPosition(finalPosition)) {
// if position is not 0, 0,
if (!(position.x === 0 && position.y === 0)) {
if (position.x === finalPosition.x && position.y === finalPosition.y) {
// if it is final Position, return code for it
return getNextCode(prevCode);
} else {
// if not store nextCode to prev code
prevCode = getNextCode(prevCode);
}
}
}
return 0;
}

0 comments on commit 53bb66f

Please sign in to comment.