Alma Andersson

What it takes to book a campsite in the US

TLDR;

I built a Python application that checks the availability of specific campsites listed at Recration.gov, and if there is anything available, it sends you an email. The code can be found here.

Intro

A few weeks ago, me and some friends were eating popcorn at a bar in Oakland (genius concept); in addition to the usual banter, we ventured into the domain of hiking and camping. Especially, how hard it is to get a spot at the campsites in the more “popular” recreational areas (e.g., Yosemite). I am yet to figure out exactly what laws that apply and where you can (legally) camp, but I’m not stoked about getting shot for trespassing, so going to a designated campsite seems like the safer option. However, there are some issues with this, which my friends explained as follows:

One of my friends also made the following statement:

All the engineers in SF know how to code, so they just write bots that book all the campsites for them, meaning ’normal’ people never have a chance to book anything.

Now, I’m not a software engineer and I very rarely write code to interact with the web, but emboldened by a few drinks I volunteered to assemble something that would help us in the battle of these coveted campsites. So that’s, in a nutshell, how I decided to write a bot of our own.

Project

Initial objective: Create a bot that will scrape the web for available campsites and notify the stakeholders if anything becomes/is available.

After thinking a bit more about the initial objective and looking into the resources, a more specific objective and workflow emerged:

Having established what features I needed out and formalized what I actually wanted to create, it was just to start building. Also, I decided to write this in Python, because that’s the language I’m most comfortable with - and I wanted this to be a small project not involving learning a new syntax/language.

The very first thing I had to do was to understand how to query Recration.gov for availability. They have an API with documentation, but this was honestly not very helpful. Browsing other github projects with similar objectives. By doing so, I realized that I could get all the information that I needed by sending a GET request to their website. The tricky part was to understand exactly how this worked, after some time, I managed to wrap my had around it. I could get what I wanted by using the get function from the requests module to send the GET request, passing it the follwoing arguments:

A successful query would return status 200 and a response object with a .json method, from which a dictionary containing all the relevant information can be extracted.

A brief comment on camping lingo:

a campground is a larger area that holds multiple campsites. At Recretion.gov you search for a campground and get availability for the different campsites within that campground.

The dictionary obtained from our (hopefully) successful query, will hold different layers of information, but what’s most interesting to us the data accessed using the campsites key; this is a new dictionary with an entry for each campsite within the queried campground; among an abundance of information, the availability is listed for all dates in the specified month. So this is exactly what we are looking for. The only additional implementations was a loop over all the months of interest and filtering to only include dates where sites are available.

I’m using a config file (YAML) to specify the some of the parameters to this application, such as the months of interest (one could do specific dates, but I found it more convenient to just say which month you want to check for dates since we want as many options as possible). I found a nice package called dateparser to help with parsing of the dates, using this allows for a more sloppy and more human friendly entry of dates.

Finally, the availability information is converted to an HTML text output which can be saved as either an html file or added as the body of an email. HTML was the first ‘programming’ language I learned, unfortunately my skills peaked at 12y/o… so all I did was to convert this to a very basic table listing the campsites for each campground together with the available dates. For each campsite there’s also a direct link to the booking page. Something like:

Campground: Point Reyes National Seashore Campground

    Campsite: BOAT A, 1-6 people
      2023-03-01
      2023-03-02
      2023-03-03
      2023-03-04
    Campsite: BOAT B, 7-14 people
      2023-03-01
      2023-03-02
      2023-03-03
      ...

Now, one of the previously listed criteria was that it should easy to add/remove campgrounds, and what’s more, I wanted different people to be able to make these changes - several people are in on this, and I don’t want every single change to go through me (that’d be a massive bottleneck). The solution to this was to create a public Google Sheets document, where anyone we shared the link with could add/remove campgrounds as we went along. This was also very easy to implement, I used the subprocess module’s run function to make a curl query, more specifically:

curl -L https://docs.google.com/spreadsheets/d/SHEET_ID/export?format=csv&gid=GID --output TEMPFILE

Where SHEET_ID is the ID of your sheet id and GID is the tab id, you can grab both of these from the URL of your sheet. I found this stackoverflow post quite helpful.

As for how updates should be shared with those of us who are in on this “scheme”, I settled on something that feels as old as stone tablets: email. Everyone checks their email, we already have apps for it, and it’s easy to set up. I’ll loop back to the email feature soon - but exactly how this would be implemented was more or less completely dictated by my choice of hosting service, so let’s say something about how that first.

I decided to host this at pythonanywhere.com which was affordable and seemed to allow enough customization. They also had a “Tasks” feature where you could launch recurring tasks just like a cronjob. Being cheap, I first went for the free subscription plan - and this made me sign up for mailgun.com as a service for email. This site was whitelisted for outgoing requests and didn’t require loads of configurations to be made (I’m looking at you Gmail…). Now, in the end I went for a paid subscription of pythonanywhere.com, but I stayed with mailgun anyway. Why? Well honestly, they had a really simple API, for real - to send an email, all you need is your API key and the follwoing function:

def send_simple_message():
    return requests.post(
        "https://api.mailgun.net/v3/YOUR_DOMAIN_NAME/messages",
        auth=("api", "YOUR_API_KEY"),
        data={"from": "Excited User <mailgun@YOUR_DOMAIN_NAME>",
              "to": ["bar@example.com", "YOU@YOUR_DOMAIN_NAME"],
              "subject": "Hello",
              "text": "Testing some Mailgun awesomness!"})

I’m not a complete idiot so I’m listing me as the receiver (to) of all emails while the other people in the group are bcc:ed. We all know each other, but as this grows it’s nice to not expose everyone’s email.

Now, what I didn’t register immediately with mailgun.com was that you needed an external domain to send your emails through (to avoid spammers… I think?). Fortunately, I had this one (differentiable.net) to use. It required me to setup some DNS configurations, but following their guide it took me at most ~20min.

To avoid redundant emails with the same availability information being sent over and over again, I opted for a semi-lazy solution. I’m saving the md5sum of the latest email (content) query, and if the result from a new query is the same - no email is sent. Evidently, this means that any change would prompt a new email, additions of campsites as well as campsites being booked. The former is probably the most interesting, but filtering for this would require a bit more work (maybe for the future) and it’s kind of good to know when spots begin to disappear.

Ok, almost there. I wrapped the workflow into a CLI where the two input arguments consist of the paths to each of two config files (YAML). Without getting into the nitty gritty details, one of these config files (static) contains emails to the people who should receive the updates, API keys etc., this information never really changes and I don’t keep it in a public space. The other config file (dynamic) contains paths to the output directory as well as the dates that should be queried; this I keep at github, for reasons that soon will become clear.

Finally, I wrote a bash script that runs the CLI application every hour (by using the Task feature at pythonanywhere.com). However, in the bash script, before running the ‘campsite checker’ (the incredibly creative name of this bot) I pull the source code from its github repository. Hence, any updates/changes to the code or the dynamic design file will be integrated in the next query.

So that’s about it. You can find the code here, although with limited documentation as of now.

In total I’d estimate that this project took somewhere between 5-8h of my time. It was a fun exercise in web interaction and made me somewhat inclined to look into JavaScript and React. Maybe for the next project?

#Random #Outdoor #Hikes #Python