I love Jack’s Gelato. Unfortunately, their website’s menu sucks. So I fixed it.
Why the existing menu sucks
The most noticeable issue of the current website is that it takes 5.8 seconds to display the menu. To add insult to injury, the text “Can take a few seconds to load..” is shown whilst you wait.
data:image/s3,"s3://crabby-images/e88bf/e88bff42636feea1e100daf98f1d661af7a0c931" alt="A screenshot of Jack's Gelato menu website."
Figure 1: A screenshot of Jack’s Gelato menu website.
Using Firefox’s web inspector tools, we can trace the network requests made to load the website HTML to attempt to diagnose the cause of the slow loading times.
data:image/s3,"s3://crabby-images/d7c24/d7c245ec914ac42672cf30eb326faca167b4de28" alt="A screenshot of the Firefox network inspector for the page load."
Figure 2: A screenshot of the Firefox network inspector for the page load.
This shows that the dominant factor in menu latency is that it is served as an embedded Google Doc through a Wix plugin. This plugin spends over a second interacting with its developer’s website, then 277ms retrieving a placeholder document to show before it starts retrieving the actual menu document. Finally, it retrieves the menu document for a total of 5.8 seconds of menu latency.
data:image/s3,"s3://crabby-images/febd9/febd96d0d15c8c85d73a80d96961189af3f9087e" alt="A screenshot of the placeholder document, retrieved by blocking request."
Figure 3: A screenshot of the placeholder document, retrieved by blocking request.
On top of the slow load times, the website transmits 9.5MB zipped of resources 1. In contrast, the gzipped text content of a representative menu is 570 bytes 2. On my pay-as-you-go data plan charging 10p per MB 3, it costs nearly one pound to load the menu website. Whilst much of this blame can be attributed to price gouging by mobile service providers, it does strongly motivate making a lighter and faster website.
How I made it better
Tl;dr (full write-up coming soon):
- Wrote a Python package which:
- Extracts the menu Google Doc URL using Selenium
- Retrieves a text representation of the menu using the Google Docs API
- Parses the text representation with a finite-state machine 4
- Generates a markdown document from the parsed menu
- Wrote a cron job in GitHub Actions which runs the package at 10:20am every day to retrieve the menu into an artefact repository
- Wrote another cron job which deploys the generated markdown menu artefacts onto my statically hosted website on GitHub Pages (for free) at 10:25am every day
My version of the website takes 147ms to load (3% of the original time), and transmits only 9.52kB of data (0.8% of the original size, 0.7p cost in pay-as-you-go data). By omitting the favicon, inlining the CSS, and eliding the inlined SVG icons this could likely be reduced to ~2kB and 40ms – but for now I am happy with the performance.
data:image/s3,"s3://crabby-images/4c6bb/4c6bb50bfa5c64450385e606ce2faa0c26178e77" alt="A screenshot of the Firefox network inspector for the page load."
Figure 4: A screenshot of the Firefox network inspector for the page load.
Please don’t sue me disclaimer
All copyright and intellectual property of the menu contents belongs to Jack’s Gelato.
I will comply with any requests to take this website down, but I hope it never comes to that. For me, this is just a fun different way to access the menu, as it makes it a bit easier/faster to work out what the flavours are each day.
Additionally, their website runs no advertising so this is not depriving them of profit in that respect.
1.02MB of HTML, 9.5MB total including JavaScript and other resources. ↩︎
This means the information a customer cares about, which icecream is available that day, is transmitted at approximately \(\frac{570 \times 8}{5.8} \approx 786\) bits per second, 0.00314% of the lower-bound average 4G bandwidth in the UK. ↩︎
This by identity could be expressed as a regular expression, but the finite-state machine is a little more readable and hence maintainable when the menu format inevitably changes ↩︎