Visualising Sydney bus congestion with Marey charts

As a side project I built Sydney Transit Graph – a visualisation of real-time Sydney bus congestion. Each line on the graph (a Marey chart) represents a bus completing its route (percentage) over time.

Head to on a laptop/desktop and have a play! It’s open source on Github.

The web app displays a map side by side with a Marey diagram. When you move your mouse across the horizontal (time) axis of the diagram, the locations of the buses are displayed at that point in time. The colour of the line represents the occupancy (warmer colours -> higher occupancy) of the service.

The 370 bus has quite the reputation! “Is the 370 the worst bus route in Sydney?”

Sydney Transit Graph uses a Marey diagram to view bus congestion in Sydney – showing the current (and historial) location of every bus.

Clicking on a line on Sydney Transit Graph highlights the position of that particular bus on the map.

The slope of the line indicates how fast the bus is going (distance verses time). A shallow line can be read as bus congestion.

Long headways

A large vertical gap between lines indicate a large distance gap between buses, and a large horizontal gap means a large time gap. Usually these are equally big or small!

In this example, there is a bus heading north-west that doesn’t have a service behind it. More people accumulate at the bus stops in that time, leading to the occupancy of the next bus being higher (orange means higher occupancy). The next bus after that also appears delayed, leading to…

Bus bunching

When the lines are bunched together, there is bus bunching – the phenomenon of multiple buses on the same route arriving at the same time (see the bus bunching Wikipedia article for an explanation on why this occurs)

Why does bus congestion matter?

Less bus congestion results in faster trip times, which in turn reduces labour costs, decreases bus headways (or reduces need for new buses) and improves passenger throughput.

Possible future work

Further analysis that would be interesting would be:

  • Correlating congestion hotspots with OpenStreetMap lane data to better inform bus lane implementation policy
  • Analysis of pre/post COVID-19 transit patterns
  • Including train data in the pipeline (it’s the same API)
  • Improve interpolation of routes on the frontend so it’s not just a straight line average between dots (interpolating along the route instead)

I’m not sure if I’ll have the time to get to this, but please let me know if you’re interested in analysis with this tool! I’ve been busy with life and other side projects so I haven’t made updates in a little while.

How it works

The web app is built with vanilla JS (TypeScript), p5.js for the graph, and Mapbox GL JS for the map (which uses OpenStreetMap data).

Data is sourced from Transport for NSW via OpenData. Realtime data is ingested every few seconds via an service running on AWS EC2 to store Protocol Buffer files on AWS S3.

When requested via an API, a service on AWS Lambda retrives these Protocol Buffer files and computes the trip progress and occupancy of every bus in a specified time window.

I adjustably quantise the number of Protocol Buffers requested (and thus the number of results) via looking up available timestamps.

Trip progress is calculated by comparing positions against bus route shapes. These route shapes have been precomputed from GTFS database dumps (via Sqlite) and stored on AWS S3 as JSON.

It’s definitely not the cleanest or most efficient architecture, however I haven’t had the time to make a big refactor recently.

Known issues

  • There are some line rendering issues in Safari

Prior art

  • Visualizing MBTA Data: An interactive exploration of Boston’s subway systemMike Barry and Brian Card:


Contributions are very welcome! If you have ideas on a project or analysis I’d love to hear. Feel free to contact me at [email protected] or leave a comment.

2 responses to “Visualising Sydney bus congestion with Marey charts”

  1. Interesting visualisation, I’m not personally familiar with the TfNSW data set but does it highlight when there’s no real time data for a bus on a route? One of the things i’ve noticed when taking buses is that if the opal real-time app doesn’t show the bus it’ll show up just before the next scheduled one and then there’ll be the bus bunching as visualised.

    • Thanks for your comment!

      > does it highlight when there’s no real time data for a bus on a route?

      From memory there is a `realtime: boolean` attribute on each bus object and it would be missing location data – so I don’t believe those buses are displayed (or highlighted) on the map.

      If they were, the line would be a very consistent slope; the “reference” timing for the bus route.

      > it’ll show up just before the next scheduled one

      Interesting, I haven’t noticed this phenomenon before. I’ll pay closer attention!

Leave a Reply

Your email address will not be published.