DEV Community

John Eliud Odhiambo
John Eliud Odhiambo

Posted on

FETCHING ADVENT OF CODE INPUT DYNAMICALLY IN GO

Advent of Code is a fun way for programmers to test and improve their problem-solving skills. While solving the puzzles, you might want to automate the fetching of your personalized puzzle input directly using its URL instead of copying the input to a text file that will be available locally. However, trying to access the input URL using a simple HTTP request, results in the message below:

Puzzle inputs differ by user. Please log in to get your puzzle input.

This article explains why this happens and how to correctly fetch your inputs dynamically using Go programming language.

The Problem: Why Can't We Fetch The Input Directly?

Advent of Code requires you to log in to access your personalized puzzle inputs. When you log in through the browser, Advent of Code sets a session cookie in your browser. This cookie is used to identify your account and provide your unique input.

If your HTTP requests don’t include this session cookie, the Advent of Code server cannot recognize you as a logged-in user, hence the error message.

Solution: Using the Session Cookie in HTTP Requests

We must include the session cookie in our HTTP requests to fetch the puzzle input. Here is a step-by-step guideline:

  • Log in to Advent of Code.

  • Open your browser's Developer Tools (Press F12 key) and navigate to the Network tab.

  • Refresh the Advent of Code page and look for the cookie header in the request headers.

Screenshot showing developers tools

  • Extract the value of the session cookie.

Screenshot showing developers tools session cookie

NOTE: It's important to keep your session cookie a secret since someone else can access your Advent of Code account if they get access to it.

Code To Fetch The Input

Below is a simple program we will use to fetch our puzzle input dynamically:

  • Setting Up The Base URL

We start by defining the base URL for fetching inputs and creating a function to read the input for a specific day.

const baseURL = "https://adventofcode.com/2024/day/%s/input"

func readInput(day string) {
    url := fmt.Sprintf(baseURL, day)
    fmt.Println(url)
}
Enter fullscreen mode Exit fullscreen mode
  • Creating The HTTP Request

Next, we create an HTTP request and include the session cookie.

client := &http.Client{}
    req, err := http.NewRequest("GET", url, nil)
    if err != nil {
        fmt.Printf("Error creating HTTP request: %v\n", err)
        return
    }

    // Add the session cookie
    req.Header.Add("Cookie", "session=[YOUR_SESSION_TOKEN]")
Enter fullscreen mode Exit fullscreen mode

http.NewRequest: Creates an HTTP GET request for the input URL.

req.Header.Add: Adds a header to the request with the session token for authentication. (Replace [YOUR_SESSION_TOKEN] with your actual token).

  • Sending The Request And Handling The Response

Now we send the HTTP request and read the server's response.

resp, err := client.Do(req)
    if err != nil {
        fmt.Printf("Error making HTTP request: %v\n", err)
        return
    }
    defer resp.Body.Close()

    if resp.StatusCode != http.StatusOK {
        fmt.Printf("Unexpected HTTP status: %d\n", resp.StatusCode)
        return
    }
Enter fullscreen mode Exit fullscreen mode

client.Do(req): Sends the HTTP request and stores the response.

defer resp.Body.Close(): Ensures the response body is closed after reading.

resp.StatusCode: Checks the HTTP status code. A code other than 200 indicates an error.

  • Reading And Printing The Input

Finally, we read the response body and print the puzzle input.

body, err := io.ReadAll(resp.Body)
    if err != nil {
        fmt.Printf("Error reading response body: %v\n", err)
        return
    }

    fmt.Println(string(body))
Enter fullscreen mode Exit fullscreen mode

io.ReadAll(resp.Body): Reads the response body.

string(body): Converts the body from a slice of bytes to a string for easy display.

  • Defining The Main Function

We invoke the readInput function from the main function to fetch the input for day 1.

func main() {
    readInput("1") // Fetches input puzzle for day 1
}
Enter fullscreen mode Exit fullscreen mode

Enhancing Security

Hardcoding the session token in our code isn’t safe. Instead, we should store it as an environment variable using the steps below:

  1. Export the session token using the terminal:
export AOC_SESSION="[YOUR_SESSION_TOKEN]"
Enter fullscreen mode Exit fullscreen mode
  1. Modify the code to read the session token from the environment variable. (Ensure to have "os" among your imports):
req.Header.Add("Cookie", "session="+os.Getenv("AOC_SESSION"))
Enter fullscreen mode Exit fullscreen mode

This helps, the session token stay outside the source code, reducing the risk of accidental exposure.

  • Full Program Code

Here's the complete program for reference:

package main

import (
    "fmt"
    "io"
    "net/http"
    "os"
)

const baseURL = "https://adventofcode.com/2024/day/%s/input"

func readInput(day string) {
    url := fmt.Sprintf(baseURL, day)

    client := &http.Client{}
    req, err := http.NewRequest("GET", url, nil)
    if err != nil {
        fmt.Printf("Error creating HTTP request: %v\n", err)
        return
    }

    // Use session token from environment variable
    req.Header.Add("Cookie", "session="+os.Getenv("AOC_SESSION"))

    resp, err := client.Do(req)
    if err != nil {
        fmt.Printf("Error making HTTP request: %v\n", err)
        return
    }
    defer resp.Body.Close()

    if resp.StatusCode != http.StatusOK {
        fmt.Printf("Unexpected HTTP status: %d\n", resp.StatusCode)
        return
    }

    body, err := io.ReadAll(resp.Body)
    if err != nil {
        fmt.Printf("Error reading response body: %v\n", err)
        return
    }

    fmt.Println(string(body))
}

func main() {
    readInput("1")
}
Enter fullscreen mode Exit fullscreen mode
  • Output

Screenshot showing output

Things To Keep In Mind

  • Session Expiry: Session tokens may expire after a while. If you encounter issues, log in again and retrieve a fresh token.

  • Privacy: Never share your session token publicly, including in blog posts or GitHub repositories.

Conclusion

You can dynamically fetch your Advent of Code inputs by including your session cookie in HTTP requests.

Feel free to share your tips or ask questions in the comment section. Happy coding, and good luck with Advent of Code 2024!

Top comments (2)

Collapse
 
ferdi_code profile image
Ferdynand Odhiambo

Great insight. Is it possible to use .env file to store the token when committing your work to github or we'll use .gitingnore since the token expire after while and we'll have to fetch a new one?

Collapse
 
johneliud profile image
John Eliud Odhiambo • Edited

Hello Ferdynand. Yes, you can store the token in a .env file. However, never push the .env file to GitHub. Instead, use a .gitignore file to prevent your version control software from tracking it. The session cookie might expire after a while, and you can always fetch a new one.