Skip to content

Latest commit

 

History

History
354 lines (251 loc) · 13.1 KB

advent-of-code-day-1-using-types.md

File metadata and controls

354 lines (251 loc) · 13.1 KB

This year I decided to take on the Advent of Code challenge along with some of my colleagues and it took me only a day to take on an interesting challenge of my own. You can look at my progress here: AGalabov/advent-of-code-2021

About the problem

Advent of Code is an yearly challenge counting down the days until Christmas. Every day participants are given two "puzzles" to solve. You only need to get to the right solution, so you are free to use any language, framework or technology. I decided that since this is my first go at the challenge, I'd stick to what I know best - Typescript and occasionally C++.

Day 1, puzzle 1

The first puzzle came up. There was a whole narrative created that would be followed throughout the event, but essentially the task could be described as simple as:

Given an array of whole positive numbers, find how many times did the numbers rise compared to their previous one. For example, given the array:

199 200 208 210 200 207 240 269 260 263

the desired outcome would be 7, the numbers that we counted being 200 208 210 207 240 269 263.

As expected from day 1 this was not a hard task. The solution I came up with took no more than a few minutes and it gave me the expected results both on sample data and on the actual input provided. It looked like this:

input
  .map((x, i) => (i > 0 ? Number(x > input[i - 1]) : 0))
  .reduce((x, y) => x + y);

Then I went on and solved the other puzzle for the day and that was that. Or so I thought...

Typescript's Type system

The day after, me and a colleague started discussing the event. He mentioned that, knowing Typescript is in fact Turing compete, it is fully possible to solve this puzzle using the type system only. At first I was confused, but then he explained his approach to me, but also that he didn't manage to finish it because of the depth limit TS enforces (as of TS 4.5 it is 1000). So that's when I decided it's game time!

The solution

What I wanted to achieve

When it comes to coding something that you are not completely sure how to implement, I find it easier if you start with the interface. So I asked myself

"What do I want to achieve?"

type Sample = [199, 200, 208, 210, 200, 207, 240, 269, 260, 263];

type Solution = NumberOfIncreases<Sample>; // Solution should be of the type 7 (the exact number 7)

Building blocks

To start with, we need a type representation of a number that would allow us to do simple arithmetics. Looking at the possible types that Typescript provides, we have simple values like boolean, number, they just won't make it - we cannot expand them. We need something that can keep multiple values (like an array) but makes a difference between having 2,3 or 10 values. Enter tuples!

A tuple is essentially an array but one of a definite size, where elements at particular positions have to be of a definite type. For example type StringAndNumber = [string, number]; defines a type. Any variables of this would type need to be an array of exactly 2 elements, first of which being a string, and the second - a number.

The important thing about tuples is that, as every array, there is a length property that can be accessed.

type Length = StringAndNumber['length']; // Length would be of type 2

To generalize it we can make Length accept a generic argument:

type Length<T extends any[]> = T extends { length: infer L }
  ? L extends number
    ? L
    : Length<[]>
  : Length<[]>;

What that code does is as simple as:

  1. Check if the provided type T has a property length
  2. If it does and it is a number - return it (L)
  3. If not - return the length of an empty array (or by transitivity - 0)

So now we know that tuples can give us the fundamental building block for numbers. So let's define some types:

type Tuple = any[];
type Zero = [];

For what it's worth, a tuple is still an array of elements. Since we only care about their length we can go with any. In any other case, please avoid using any and instead provide meaningful types in your code.

We can revisit our Length definition now to make it a little more semantically appealing:

type Length<T extends Tuple> = T extends { length: infer L }
  ? L extends number
    ? L
    : Length<Zero>
  : Length<Zero>;

So now we have our representation of numbers. But how do we actually "represent" numbers as tuples. Again if we think about the usage first, we would want something like BuildTuple<5> to create a tuple like [any, any, any, any, any]. We can do this recursively:

type BuildTuple<N extends number, T extends Tuple = Zero> = T extends {
  length: N;
}
  ? T
  : BuildTuple<N, [...T, any]>;

Again this code might look a little scary to someone but it can be read as:

  1. We take a number N and an accumulated value T (which by default is our Zero)
  2. We check if the length property of T is exactly the desired number N
  3. If it is - then T is our tuple representation
  4. If not - we recursively call BuildTuple much like a function with the new accumulated value being T with an additional any at the end

Just to check that everything works as expected, let's apply Length on a type created from BuildTuple

type Five = Length<BuildTuple<5>>; // Five is actually of type 5.

Limitations

As I already mentioned there are recursion depth limitation when it comes to Typescript:

type UnderLimit = BuildTuple<999>; // this would be infered correctly
type AboveLimit = BuildTuple<1000>; // this one results in "Type instantiation is excessively deep and possibly infinite"

Arithmetics:

Now that we have our concept of Zero and can create any other number as a tuple representation, how do we do arithmetics.

Addition

Let's start with defining PlusOne:

type PlusOne<N extends number> = Length<[...BuildTuple<N>, any]>;

This is self explanatory:

  1. Given a number N we create a tuple from it using BuildTuple
  2. We then create another tuple, expanding the current tuple and adding one more type element (any).
  3. Finally we use Length so that we go back to a constant number type.

As a result we have:

type Six = PlusOne<5>; // Six is of type 6

The way to define Plus for two numbers is pretty obvious now:

type Plus<A extends number, B extends number> = Length<
  [...BuildTuple<A>, ...BuildTuple<B>]
>;
type Twelve = Plus<4, 8>; // Twelve is of type 12

Subtraction

Doing the opposite we can define MinusOne:

type MinusOne<N extends number, T = BuildTuple<N>> = T extends Zero
  ? Length<Zero>
  : T extends [any, ...infer Rest]
  ? Length<Rest>
  : Length<Zero>;

What we do here is the following:

  1. Given a number N we create a tuple from it using BuildTuple and save it as T
  2. We check if T is Zero and if so we return Length<Zero> (which is 0)
  3. If not, we check if T can be represented as a single type any and some values Rest
  4. If we can, then Length<Rest> is the result of the subtraction (we have removed one element)
  5. If not - then we have a 0

As a result we have:

type Thirteen = MinusOne<14>; // Thirteen is of type 13

And just like with Plus it's quite easy to define a Minus now:

type Minus<
  A extends number,
  B extends number,
  T = BuildTuple<A>
> = T extends Zero
  ? Length<Zero>
  : T extends [...infer Rest, ...BuildTuple<B>] // just like with the any, we just define and expand a tuple of the size to subtract
  ? Length<Rest>
  : Length<Zero>;

type Seven = Minus<13, 6>; // Seven is of type 7

Comparisons

In order to find the number of increases we would need to be able to compare two numbers. Let's do it:

type GreaterThan<
  A extends number,
  B extends number
> = BuildTuple<B> extends Zero
  ? BuildTuple<A> extends Zero
    ? false
    : true
  : GreaterThan<MinusOne<A>, MinusOne<B>>;

Let's go line by line:

  1. We create a tuple from B and check if it is Zero
  2. If it is - we create one from A as well and check it against Zero.
  3. If it is - A and B are equal, hence we return false
  4. If A isn't, then A is greater than B and we return true
  5. If B is not Zero then we recursively call GreaterThan subtracting one from both A and B

The results is as expected:

type Greater = GreaterThan<5, 3>; // Greater is of type true
type Equal = GreaterThan<5, 5>; // Equal is of type false
type Smaller = GreaterThan<3, 5>; // Smaller is of type false

The actual solution

Let's take a look at it from a developer's standpoint and try to find a recursive solution (since this is how TS works):

function numberOfIncreases(
  numbers: number[],
  lastNumber: number | null = null,
  accumulator: number = 0
): number {
  if (numbers.length === 0) {
    return accumulator;
  }

  const [current, ...rest] = numbers;

  if (lastNumber && current > lastNumber) {
    return numberOfIncreases(rest, current, accumulator + 1);
  }

  return numberOfIncreases(rest, current, accumulator);
}

numberOfIncreases(input);
  1. The bottom of our recursion is when we have gone through all elements. In that case we return the accumulated value.
  2. If we have a lastNumber (any call apart from the first one) and the current is greater, we recursively call the function on the rest of the numbers with an increased accumulator.
  3. If not, then we just call the function recursively on rest without increasing the accumulator.

Now let's "translate" this logic into a type definition:

type NumberOfIncreases<
  Input extends number[],
  Previous extends number | null = null,
  Accumulator extends number = Length<Zero>
> = Input extends []
  ? Accumulator
  : Input extends [infer Curr, ...infer Rest]
  ? Previous extends number
    ? GreaterThan<Curr, Previous> extends true
      ? NumberOfIncreases<Rest, Curr, PlusOne<Accumulator>>
      : NumberOfIncreases<Rest, Curr, Accumulator>
    : NumberOfIncreases<Rest, Curr, Accumulator>
  : Accumulator;

Now sadly the direct "translation" does not work entirely. Typescript infers Curr as unknown instead of as number. To fix this we could:

  1. Add a check like Curr extends number
  2. Use Input[0] instead of Curr to have proper inference

We could however simplify the logic by extracting some of it. Let's extract the logic behind getting the rest elements of an array:

type RestFromArray<T extends Tuple> = T extends [any, ...infer Rest]
  ? Rest
  : Zero;

Now if we create a type using RestFromArray<[1, 2, 3]> it would result in [2, 3].

Now the final implementation would then look like this:

type NumberOfIncreases<
  Input extends number[],
  Previous extends number | null = null,
  Accumulator extends number = Length<Zero>
> = Input extends []
  ? Accumulator
  : Previous extends number
  ? GreaterThan<Input[0], Previous> extends true
    ? NumberOfIncreases<RestFromArray<Input>, Input[0], PlusOne<Accumulator>>
    : NumberOfIncreases<RestFromArray<Input>, Input[0], Accumulator>
  : NumberOfIncreases<RestFromArray<Input>, Input[0], Accumulator>;

This is a very good example how actual logic can be written with types.

And now after all of this we get to the moment of truth. Let's test it out? Are you excited? Let's go:

type Sample = [199, 200, 208, 210, 200, 207, 240, 269, 260, 263];

type Solution = NumberOfIncreases<Sample>;

IT WORKS! Solution is correctly inferred as the type 7.

Stress test

Now that the puzzle has been solved using the sample input I wanted to try how far it can go. I entered the full input from my assignment. It contains 2000 numbers ranging from 174 to 4618... THen immediately tsc starts to throw "Type instantiation is excessively deep and possibly infinite" error once again. But actually that shouldn't come as a surprise to anyone right now. We already discovered using the current implementation even running BuildTuple<1000> reaches the maximum depth of calculation. So we cannot expect to create one for 1938, 2002, 4618 (numbers from the input) and so on and then make computations on top of them.

I then tried using different subsets of my input data to see what the limits are. The best I got was actually still very impressive:

type Full = [174, ..., 912] // a total of 433 numbers

type Solution = NumberOfIncreases<Full> // resulting in 271

Conclusion

We did manage to implement an algorithm that uses types only to solve an Advent of Code level puzzle.

Is it useful? - No!
Is it something that you need to do in your life? - Probably no!
Was it fun? - For me, yes!
Was it worth it? - YES!
Will I try this for the next puzzles? - YES!

Takeaway points

In the process of implementing the building blocks, arithmetics, comparisons and the actual algorithm:

  1. I found out some new capabilities of Typescript
  2. I reminded myself of some math fundamentals
  3. I wrote my first and hopefully not last blog post
  4. I found myself a new hobby

You can find the full code and check out my progress with the other puzzles on AGalabov/advent-of-code-2021/type-system