diff --git a/content/posts/kreuzungen.md b/content/posts/kreuzungen.md new file mode 100644 index 0000000..18fc523 --- /dev/null +++ b/content/posts/kreuzungen.md @@ -0,0 +1,1533 @@ +--- +types: posts +title: "Kreuzungen 🗺️" +subtitle: "Where open geospatial data and cycling intersect." +date: 2024-08-11T21:41:55+01:00 +lastmod: 2024-08-11T21:41:55+01:00 +draft: false +description: "Where open geospatial data and cycling intersect." + +tags: [] +categories: [] +series: [] + +hiddenFromHomePage: false +hiddenFromSearch: false + +featuredImage: "/media/kreuzungen/oil_crossings.jpeg" +featuredImagePreview: "" + +toc: + enable: true + auto: false + + +math: true +lightgallery: false +license: "" +--- + + +{{< admonition type=abstract title="Introducing Kreuzungen 🗺️" >}} +I recently made a website that reveals rivers and streams encountered on recent cycling or hiking adventures. +![Screenshot of https://kreuzungen.world](/media/kreuzungen/screenshot_frame.png) +{{% center %}} +[https://kreuzungen.world](https://kreuzungen.world) +{{% /center %}} + +The site is powered by OpenStreetMap data and open source technologies. + +All source code is available in this [github repository](https://github.com/01100100/kreuzungen) + +{{< /admonition >}} + +## Introduction + +Last year I embarked on a bike ride from a [geo-spatial-data-conference](https://spatial-data-science-conference.com/) in London to my home in Berlin. + +I used [Komoot](https://www.komoot.com) to plan, navigate and record my journey. I set my start point to the Royal Albert hall in London and my end point to Brandenburg Tor in Berlin, with intermediate points being the international ferry ports of Harwich and Hook of Holland. I choose "Road cycling" as my preferred style of riding and hit the "Let's go" button. It was that easy! I am a huge fan of the [Komoot](https://www.komoot.com) app and highly recommend it. + +![Komoot Screenshot](/media/kreuzungen/komoot_screenshot.png) + +During my multi-day ride, crossing various rivers, a thought struck me: + +{{% center %}} +***How many waterways have I crossed and what are they called?*** +{{% /center %}} + +Realizing that I had recorded data of the ride, had the great asset of open street maps at my fingertips, and the geo-spatial-data know how, I put a plan together... + +{{< admonition type=tip title="The Plan" >}} + +{{% center %}} +**To develop a tool that could be fed with a .gpx recording. It should list all the waterways crossed en route and display a map with the journey and highlight all crossed waterways.** + +{{% /center %}} +{{< /admonition >}} + +As a geospatial data engineer, and a lover of long cycles in really remote places, this challenge doesn't sound too hard. + +Little did I know that this project would turn out to be much more interesting than first expected, just like the bike journey. I didn't know where exactly the project would take me, but I had a goal and simply started out ready to adapt. + +{{< admonition type=info title=Naming >}} +The word "Kreuzungen" is German for "crossings" or "intersections"... + +{{% center %}} +[https://de.wiktionary.org/wiki/Kreuzung](https://de.wiktionary.org/wiki/Kreuzung) +{{% /center %}} +{{< /admonition >}} + +## Geospatial Data 101 + +Geospatial data, in its simplest form, refers to information that describes an object in space. + +It consists of two parts: the "where" and the "what". The "where" part includes spatial information describing the geometry and location of an object, while the "what" part includes non-geometric properties/attributes that describe and give meaning to an object. + +### Geometry + +The spatial part is composed of different types of geometries. The primitive types of geometry are points, lines and polygons. These are the building blocks of geospatial data, and out of them we can build some amazing things. + +TODO: add popups for the examples with a map and few cases of each. + +- A Point: represents a specific location on the Earth's surface, defined by its latitude and longitude coordinates. It is often used to mark a point of interest, such as a atm, traffic light or water fountain. +- A LineSting: represents a path made up of a ordered series of connected points. A hiking path, road or a river. +- A Polygon: A polygon is a closed shape with the vertices defined by a ordered series of points. It is used to represent areas like countries, cities, or lakes. Polygons are defined by a list of coordinates that outline their boundaries. + +```goat + *-----* + * *---*-* / \ + \ / \ + *---* *-----------* +``` + +**Any geometry combined with non-geometric properties/attributes is know in the industry as a geometric feature.** + +Sometimes, a single geometry type might not be sufficient to represent a feature. For example, a river system with multiple branches cannot be accurately represented by a single LineString. In such cases, we can use multi geometry features like a MultiLineString or a MultiPolygon. + +TODO: add a image with a river system and the different LinesStrings that would make the MultiLineString + +We can group different features together to form a feature collection. Imagine you want to represent all the rivers, lakes and wells in a region. You could group them together in a feature collection using LineStrings for the rivers, Polygons for the lakes and Points for the wells. + +### An Example: Dicke Marie (Thick Mary) + +If you go into Tegel Forest in the east of Berlin, you might bump into the so called "Thick Mary". This is the name given to a really old, award winning[^dicke-marie] oak tree, estimated to be over 800 years old. + +[https://berlindoodleblog.blogspot.com/2015/02/dicke-marie.html](https://berlindoodleblog.blogspot.com/2015/02/dicke-marie.html) + +![Dicke Marie](/media/kreuzungen/dickemarie.jpg) +Let's imagine what the geospatial data would look like that describes this great tree... + +First we should define the location or the "where" of Dicke Marie. We could use words to do this, perhaps "North East of Berlin", "In Tegel" or "about 300m south of Tegel Schloss" would work. However we can do much better if we define with the position with a `POINT` geometry. We could do this with a a single set of co-ordinates `(52.5935770, 13.2649068)` representing the middle of the tree trunk. + +These geometric data types are much more powerful then the wordy descriptions. Defining the position of thick Mary with a `POINT` fixes it to a set position on a grid of the world, and the value can be quickly and efficiently compared to other geospatial data types to answer questions like.. "How far is this point east of point x" "Does this point lie within polygon z?". Ohhh yeahhhh, I am starting to feel the power of this geospatial stuff. + +Now we can add some "what" properties describe Fat Marie. Perhaps we can add the following properties: + +| Property | Value | +|----------|-------| +| type | tree | +| name | Dicke Marie | +| species | Quercus robur | +| height | 23 | +| wikipedia | [https://de.wikipedia.org/wiki/de:Dicke_Marie](https://de.wikipedia.org/wiki/de:Dicke_Marie) | + +Using the geojson data format, a Dicke Marie could be written like this + +```json +{ + "type": "Feature", + "properties": { + "name": "Dicke Marie", + "natural": "tree", + "species": "Quercus robur", + "height": "23", + "wikipedia": "de:Dicke Marie" + }, + "geometry": { + "type": "Point", + "coordinates": [ + 13.2649068, + 52.593577 + ] + } +} +``` + +In fact someone already defined Dicke Marie using geospatial data and added it to Open Street Maps + +[https://www.openstreetmap.org/node/205066401](https://www.openstreetmap.org/node/205066401) + + + + +## What is a river? + +The big question! A river is a natural flowing watercourse that typically moves towards an ocean, sea, lake, or another river. It plays a vital role in the Earth's hydrological cycle and supports various ecosystems and human activities[^river-wiki] + +There is a hour long podcast that dives into this in more depth. + + + +This a very interesting question to think about, but philosophizing about what constitutes a river doesn't exactly help me solve the problem of working out which rivers I crossed on my bike ride. A better question would be __**"Where can I get data from and how can I define waterways in geospatial data terms?"**__ + +## OpenStreetMap + +![OpenStreetMap logo](/media/kreuzungen/oslogo.svg.png) +TODO: link to OSM iceberg meme + +[OpenStreetMap](https://www.openstreetmap.org/) (OSM) is an incredible resource, akin to the Wikipedia of maps. It's a collaborative project where a community of mappers from around the world contribute to creating a free and editable map of the world. This was not possible 25 years ago, and the existence of such a resource today is a testament to the power of community and open source efforts. + +The data is freely available under an open license, and it's maintained and updated by volunteers. This means that the data is constantly being updated, reflecting changes in the real world. + +Everyone can make use of this data and create applications scaling to cover the whole world. OSM already powers most of the maps and map related services you find on the web today[^osm-services]. + +{{< admonition type=info title="Street Complete" >}} + +[StreetComplete](https://streetcomplete.app) is an Android app that allows users to easily edit and improve OpenStreetMap data without any specific knowledge of OSM tagging schemes. It asks simple questions and uses the answers to directly edit the map. It's designed for users who want to contribute to OpenStreetMap but don't have expertise in OSM. + +{{}} + +One of the key features of OSM is the ability to tag different elements with various attributes. For this project, the 'waterway' key is particularly useful [https://wiki.openstreetmap.org/wiki/Key:waterway](https://wiki.openstreetmap.org/wiki/Key:waterway) +. This key is used to describe natural or artificial water flows like rivers, streams or canals, as well as elements which control the water flow such as dams and weirs. Sounds exactly what I am looking for to work out which rivers I crossed. + +Here are examples of different waterways in OSM: + +TODO: add links to osm elements + +| Waterway Type | Description | Example | +|---------------|-------------|---------| +| [River](https://wiki.openstreetmap.org/wiki/Tag:waterway%3Driver) | A large natural waterway | The River Thames [] | +| [Stream](https://wiki.openstreetmap.org/wiki/Tag:waterway%3Dstream) | A small natural waterway | A stream in your local park | +| [Canal](https://wiki.openstreetmap.org/wiki/Tag:waterway%3Dcanal) | A man-made waterway used for transportation, irrigation, or drainage | The Suez Canal | +| [Drain](https://wiki.openstreetmap.org/wiki/Tag:waterway%3Ddrain) | A man-made waterway used to drain water | | + +### OpenStreetMaps data model + +OpenStreetMap's data model is quite simple, there are three main elements which are used to represent the entire world: + +- **Nodes**: These are individual points on the map. Each node has a latitude and longitude. For example, a node could represent a park bench or a water fountain. A node corresponds to a **POINT**. + +- **Ways**: These are ordered lists of nodes, representing a polyline on the map. Ways are used for longer features that cover more distance, like roads or rivers. A way corresponds to a **LineString**. + +- **Relations**: These are groups of nodes, ways, and other relations that define a larger entity. For example, a relation could represent a cycle route that consists of many different ways. A relation corresponds to a MultiLineString or a Polygon, depending on the nature of the relation. + +#### Tagging + +Each of these elements can have tags, which are key-value pairs used to store metadata about the element. For example, a way representing a road could have a tag `highway=residential` indicating it's a residential road. + +For more information, visit the [OpenStreetMap Elements page](https://wiki.openstreetmap.org/wiki/Elements). + +#### Waterways + +When we query OSM for waterways, we get back a line or multiline string representing the waterway in OSM format. + +There is a difference between the waterway [**way**](https://wiki.openstreetmap.org/wiki/Way) in OSM and the greater [waterway **relation**](https://wiki.openstreetmap.org/wiki/Relation:waterway). A waterway **way** is a single segment of a waterway. A waterway **relation** is a group of the smaller ways to form a larger entity, like a big river can be made from many smaller streams. + +Most "main" rivers in Europe are tagged as relations, while almost all smaller rivers and streams are represented as ways. + +You can visit [this link](https://wiki.openstreetmap.org/wiki/Waterways). It provides detailed explanations about the tags used for waterways and the properties associated. + +### Overpass API + +The OSM database is huge. [Planet.osm](https://wiki.openstreetmap.org/wiki/Planet.osm) is a single file containing all OpenStreetMap data (currently over 1902.6 gb uncompressed and growing), and it's not practical to download and process the entire dataset every time you want to work with a few features. + +OpenStreetMap offer a read-only public API with [http://overpass-api.de](http://overpass-api.de). It has a usage policy: You can safely assume that you don't disturb other users when you do less than 10,000 queries per day and download less than 1 GB data per day[1]. + +This API makes it easy to request data and work with it programmatically, and by providing a query language called Overpass QL, which is similar to SQL, which allows you to get exactly what you need from a truly vast amount of geospatial data. This is perfect for this project. + +{{< admonition type=info title="Overpass Turbo" >}} + +[Overpass Turbo](https://overpass-turbo.eu/) is a web-based data filtering tool for OpenStreetMap. It's a great tool for testing out queries because it provides a visual interface for constructing and running Overpass API queries. It also displays the results on a map, making it easy to verify the queries. + +![Load of rivers on overpass!](/media/kreuzungen/overpassturbo.png) + +// TODO: link to the query that produced this. + +{{}} + +TODO: talk about how a big part of the problem is filtering the data to as little as possible before running the computations, and you should do as much of this upstream as possible with the overpass api. + +## Ok enough theory, lets solve the problem. + +Breaking it down there are a few parts to the tool I want to make: + +1. Uploading a route. +1. Fetching river data from OSM +1. Calculating the intersections +1. Visualizing the results + +A little website could work well for this. The browser can take care of the the route uploading with the [FILE API](https://developer.mozilla.org/en-US/docs/Web/API/File_API), requesting OSM data can depend on the [FETCH API](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API), and visualizing it with the help of [MapLibre GL JS](https://maplibre.org/maplibre-gl-js/docs). Processing the route and computing intersection can be done natively in language of the browser, javascript, with high grade geospatial functionality being imported from [turf.js](https://turfjs.org/). By depending on the browser, I should be able to avoid the need of hosting any backend and reduce the complexity of the project. + +### Design + +The website should be simple and easy to use. Easier said then implemented! + +When the user navigates to [https://kreuzungen.world](https://kreuzungen.world) I want them to see a map. There should be a container with text explaining the site, and a button to upload a route from there device. When the user uploads a .gpx file, it should be displayed on the map. Then the site should fetch the waterway data in the background and compute the intersecting waterways before highlighting them on the map and listing the river names. The user should be able to explore the map and river data using intuitive interactions, there should be information shown for every river with any +interesting properties links to the upstream data source. + +### The build + +### The body of the machine + +The html should be quite simple (hopefully). Let's start basic with a input for the file handling and a div for the map. The map should take up the whole screen. Inside the map should be a couple be containers for the displaying of text information. A "info-container" will explain how to use the site, and another which is initially set to not be displayed, but will later show information from the uploaded route. + +```html + + + + + + + + + + +
+
+
+

Welcome! 🌍🚴‍♂️

+
+
+ +
+ + + +``` + + +I will use MapLibre GL JS to render the map. It's a great library that allows you to create interactive maps with vector tiles. It's based on WebGL, which means it can render maps quickly and efficiently in the browser. + +MapLibre will inject a map into the div with the id "map". The map will be centered on the [natural center of the world in Greenwich](https://en.wikipedia.org/wiki/Prime_meridian_(Greenwich)#history), and have a initial zoom level of 10. The map will use a "outdoor-sy" style from [MapTiler](https://www.maptiler.com/), a company that provides vector tiles for maps. + +### Error! DataFormat Unknown + +If you ran the above code, you would have unfortunately noticed a nice error message. In life, things don't always work out first try... That's ok, we can look into the error, and we can fix it. Here you'll see that MapLibre is expected at Geojson and not the gpx file that the user uploaded. We are very fortunate that there is a great library out there that will convert between these from MapBox. So let's Plug that in and get things up and running. + + +```typescript +export async function parseGPXToGeoJSON(GPXContents: string) { + const doc = new DOMParser().parseFromString(GPXContents, "text/xml"); + return toGeoJSON.gpx(doc); +} +``` + + +### Uploading a route + +I added a hidden [file input](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/file) to the html to facilitate the file upload. The user can then click on a button to trigger the file input, or + +The user can then select a file, and the file input will emit a change event. I added an event listener to the file input, such that the change event will call code that reads the .gpx file, displays it on the map and then processes it. + +```html + + +
+ +
+ ... +
+
+``` + +#### Why convert? + +By converting the OSM and route geographic data to geojson, I could use turf.js to do all spatial analytics in the js code, meaning the computing would be done client side and no backend would be required! + +TODO: + +### Fetching river data from OSM + +I can use the Overpass API to fetch all the waterways within a bounding box. The query I used looks like this + +``` +[out:json];(rel["waterway"]({{bbox}});way["waterway"]({{bbox}});)->._;out geom; +``` + +This has three parts to it, and aims to capture all potentially intersecting waterways, while keeping the data as small as possible. + +* `[out:json]` - specifies that the output should be in JSON format. +* `(rel["waterway"]({{bbox}});way["waterway"]({{bbox}});)->._` selects all the relations and ways tagged as "waterway" within the bounding box and combines the result. The `{{bbox}}` part is a placeholder that gets replaced with the actual bounding box coordinates. +* `out geom;` means that [the geometry of the selected elements](https://wiki.openstreetmap.org/wiki/OSM_XML#Overpass_API_out_geom) is returned. + +TODO: explain how when we have a bigger region, we might want to only look at relations and not ways. + +TODO: talk about when it would make sense to download the OSM planet dataset and then store and serve the waterways data on my own server, and when it would make sense to fetch it from the overpass api. + +Going back to my point about not needed a backend, this is only possible due to the existence of the overpass api. + +However on my webhook backend server, I am likely going to request the same river data many times for well traveled regions. Eacch subsequent request is redundant (Ignoring the temporal changes. If someone has built a new canal in Berlin and added it to OSM, since the last request of Berlin waterways, then it is worth knowing about). The idea of caching requests is a good one, however its not practical due to the requests being specific to a routes bounding box, and the fact they are unlikely to not be unique. It would be more efficient to download the OSM data once and store it on my server, accessing it willi nilli. Although this adds a burden on me, to maintain a server and keep the data up to date, it would be more efficient and faster for the user, and most importantly would limit the number of redundant requests to the overpass api. However, I don't imagine my little project will be getting that much traffic, so I will stick with the overpass api for now and deal with this if/when I get hit with some major traffic. + +consider when we are getting the citys a route crosses through, there are two ways to make a scaleable solution for the whole world: + +1. Download the OSM planet dataset and store it on a server, then query the server for the data. +2. Use the overpass api to fetch the data on the fly. + +The first option is more efficient, but requires a lot of storage and maintenance. The second option is less efficient, but is easier to set up and maintain. + +The reality is that I am a very little app with a few users, and the users are located in very few places. I can get away with using the overpass api for now, and if I get more users, I can always switch to the first option and save some load on the overpass api . + +#### Combining the LineStrings and MultilineStrings into a feature collection of single rivers + +The Overpass API returns a list of features, each representing a waterway. The features can be of type `LineString` or `MultiLineString`, depending on the complexity of the waterway. + +There are also thousands of disjoint ways that make up a single river. To make the data easier to work with (due to the performance fact that we only look for the single intersection point and skip the rest), I combined all the features with a matching name into a single feature. + +This makes it easier to iterate over the individual rivers and calculate intersections with the uploaded route. + +The one downside is the problem of common name rivers... TODO: talk about this. + +### Intersections + +TODO: introduce turf.js + +TODO: talk about converting everything to geojson + +TODO: talk about combining the features in the feature collection together to handel both ways and relations + +TODO: talk about the self-intersections problem and the hidden option of passing the `{ ignoreSelfIntersections: true }` parameter to the `lineIntersect` function. + +TODO: talk about the tests used to verify this was fixxed ^^ + +```js + +#### Let's talk about time complexity + +Finding out which rivers intersect with the uploaded route is taken care of by turf.js, using the [`booleanIntersects()`](https://github.com/Turfjs/turf/tree/master/packages/turf-boolean-intersects) function. + +```js +function filterIntersectingWaterways(waterwaysGeoJSON, routeGeoJSON) { + return waterwaysGeoJSON.features.filter((feature) => + turf.booleanIntersects(feature, routeGeoJSON) + ); +} +``` + +The `turf.booleanIntersects()` implementation uses the [sweepline-intersections](https://github.com/rowanwins/sweepline-intersections?tab=readme-ov-file#algorithm-steps) algorithm, and although it is optimized to blaze through co-ordinate pairs super fast, it has quadratic [time complexity](https://en.wikipedia.org/wiki/Time_complexity). + +$$ \mathcal{O}(n^2) $$ + + +{{< admonition type=tip open=true >}} +That's a fancy way of saying as the input data gets big (number of waterways to check for intersections), the time is takes to compute gets realllllly big. +{{< /admonition >}} + +The key to solving the problem in a "acceptable time" involves keeping the input data "somewhat" small such that device doing the computation does not melt and crash. + +TODO: talk about the long tail distibution. + +TODO: napkin math + +TODO: flash a message on the screen. + +### Visualising the results + +#### Maplibre gl + +TODO: write about slippy maps, how good maplibre is because of vector tiles, allowing for zooming in and out without pixelation and ability to render features conditionally on the client side. + +TODO: include some history of webmaps. + + +https://en.wikipedia.org/wiki/Web_mapping + +TODO: talk about the tradeoffs between vector and raster maps, and explain why vector maps are the best for this project. + +TODO: Link to and explain that even OSM are going to vector based maps this year because they are so much better. [https://blog.openstreetmap.org/2024/02/11/2024-announcing-the-year-of-the-openstreetmap-vector-maps/](https://blog.openstreetmap.org/2024/02/11/2024-announcing-the-year-of-the-openstreetmap-vector-maps/) + +### Vector tile provider + +TODO: talk about how it would be possible to generate these using open data and a machine, but it is not needed as there are many shops offering this product, in a way thats been optimally distibuted (CDN) + +eg) The complete OSM vector tile data set is >110gb https://data.maptiler.com/downloads/dataset/osm. + +https://wiki.openstreetmap.org/wiki/Vector_tiles#Providers + +TODO: Talk about maptiler and stadia maps both offering free tiers for serving vector map tiles. Maptiler give 100,000 requests per month for free. + +### Features + +TODO: talk about being happy with the look and feel of maplibre, and reusing the css for displaying info. + +#### Custom controllers + +TODO: talk about the code to make the upload and strava controller. + +#### Contain yourself + +### Killer feature: Strava integration + +TODO: explain that while some people do, most people don't know what a .gpx file is and don' +t have files laying stored on there local device. The typical cyclist/hiker would likely have activities on strava. Explain that you implemented a "connect with strava" button and wrote the code to fetch activities from strava and display them in a menu withing the app. Explain how this lets users circumvent needing to know what a .gpx file or uploading anything and leads to a streamlined experiance for the user. + +### Strava Oauth + +TODO: we want users to connect with strava and then request the activites data from Strava. give a brief explination of what a oauth flow is and explain why it is impossible to avoid situation of adding a backend to implement strava oauth. Explain that a backend is needed to hold the client secret and avoid it being in the front end and thus leaked into the world. Explain the intention to keep the backend as lightweight as possible using flask and point to the codein `src/auth.py`, and mention much could be improved. + +> All developers need to register their application before getting started. A registered application will be assigned a client ID and client secret. The secret is used for authentication and should never be shared. +- https://developers.strava.com/docs/authentication/ + +Putting the client secret in the frontend would compromise the security of the application. + +TODO: talk about using flask and cors, and serving it with waitress because security reasons. + +```python +import requests +from flask import Flask, request +from waitress import serve + +from config import get_config_values, get_logger + +### SET UP ENVIRONMENT ### +logger = get_logger() +CONFIG = get_config_values() + +app = Flask(__name__) + + +@app.after_request +def after_request(response): + response.headers.add("Access-Control-Allow-Origin", "*") + response.headers.add("Access-Control-Allow-Headers", "Content-Type,Authorization") + response.headers.add("Access-Control-Allow-Methods", "GET,PUT,POST,DELETE,OPTIONS") + return response + + +@app.route("/oauth", methods=["POST"]) +def oauth_callback(): + code = request.form.get("code") + response = requests.post( + f"{CONFIG.STRAVA_API_URL}/oauth/token", + data={ + "client_id": CONFIG.STRAVA_CLIENT_ID, + "client_secret": CONFIG.STRAVA_API_CLIENT_SECRET, + "code": code, + "grant_type": "authorization_code", + }, + ) + if response.status_code != 200: + raise Exception( + f"Error fetching access token, status code {response.status_code}" + ) + response.json() + + return response.json() + + +@app.route("/reoauth", methods=["POST"]) +def refresh_token(): + refresh_token = request.form.get("refreshToken") + response = requests.post( + f"{CONFIG.STRAVA_API_URL}/oauth/token", + data={ + "client_id": CONFIG.STRAVA_CLIENT_ID, + "client_secret": CONFIG.STRAVA_API_CLIENT_SECRET, + "refresh_token": refresh_token, + "grant_type": "refresh_token", + }, + ) + if response.status_code != 200: + raise Exception( + f"Error refreshing access token, status code {response.status_code}" + ) + response.json() + + return response.json() + + +if __name__ == "__main__": + serve(app, listen="*:8080") +``` + +TODO: talk about using this dockerfile to build the backend service + +```Dockerfile +FROM python:3.11-slim-buster + +COPY requirements.txt requirements.txt +RUN pip install -r requirements.txt + +COPY src src + +ENTRYPOINT ["python", "src/auth.py"] +``` + +TODO: talk about deploying this on fly.io using the following config + +```toml +# See https://fly.io/docs/reference/configuration/ for information about how to use this file. +# + +app = 'kreuzungen' +primary_region = 'ams' + +[http_service] + internal_port = 8080 + force_https = true + auto_stop_machines = true + auto_start_machines = true + min_machines_running = 0 + processes = ['app'] + +[[vm]] + cpu_kind = 'shared' + cpus = 1 + memory_mb = 256 +``` + +{{< admonition type=tip >}} + +Setting the `auto_stop_machines` and `auto_start_machines` options on a fly.io service will automatically turn off the machine when it's idle and turn it on whenever there is a incoming request, akin to turning off a engine at a stopped traffic light[^start-stop]. This means your only charged for whatever you need and will save some money to spend on fun activities. This completely makes sense for any non critical service that is not being used 24/7. + +[https://fly.io/docs/apps/autostart-stop/](https://fly.io/docs/apps/autostart-stop/) +{{< /admonition >}} + +``` bash +fly secrets set STRAVA_API_CLIENT_SECRET=$STRAVA_API_CLIENT_SECRET +fly secrets set STRAVA_REDIRECT_URI=$STRAVA_REDIRECT_URI +fly deploy +``` + +### Answering the community**: Make it easy to share the good stuff + +A *early-beta-tester* (you know who you are) gave me some feedback that they would like to share the site with friends. This which gave me a few ideas: + +- Have the website preview displayed nicely when shared in a messaging app or on social media by using the [Open Graph protocol](https://ogp.me/) +- Have a share-with-social-media* button. +- Create a url link to a actual activity from a user which they could send to a mate to explore. + - Encode the a users activity into a string. + - Add it to a sharable link with a url parameter. + - Parse the route and decode it on the new client. + +#### What the OPG? + +> The Open Graph protocol enables any web page to become a rich object in a social graph** + +That sounds pretty good! Basically you can add some specific `` tags to the `` of a website, and when you share a url in social-media/messaging apps will render the content and display a little preview. You know when you share a link in whatsapp and it shows a little image and a description? That's the Open Graph protocol in action. + +This was a easy one to implement after reading the spec on [https://ogp.me](https://ogp.me). I looked up the optimal image dimensions and according to *reasons* this was **1200px x 630px**. The result was added in the following code + +```html {linenos=table,hl_lines=["4-10"],linenostart=4} + + Kreuzungen 🗺️ + + + + + + + + + +``` + +At first the image was not being displayed when I shared it via whatsapp, however a [quick stackoverflowing](https://stackoverflow.com/a/39182227) explained WhatsApp only supports images less than 300kb in size so I added a compressed image. + +![Loading the url in whatsapp](/media/kreuzungen/whatsapp.gif) + +Everything worked, the internet is magic sometimes! + +{{< admonition type=tip >}} +https://opengraph.dev is a nice site to test out how the content is rendered on different platforms. +{{< /admonition >}} + +### Share button + +> Thousands of candles can be lit from a single candle, And the life of the candle will not be shortened. Happiness never decreases by being shared. - The Buddha + +I wanted to have a nice and easy way for people to share the map, because happiness is good. to implement this I created another control with the well known [share icon](https://en.wikipedia.org/wiki/Share_icon). + +{{< admonition type=note >}} + +TODO: put in the many different share icons that people are used to... talk about standardizing this [share icon](https://en.wikipedia.org/wiki/Share_icon) + +{{}} + +When the control is clicked on, it should expand and show different options for quickly sharing on different platforms using as well as having a copy button that will copy the url to the clipboard. + +The clickingaround-test showed that the copy-to-clipboard didn't feel like it was working because there was no user feedback. To fix that I implemented a message to flash on the screen and signify that the copy action has been done, it fades out after 0.5s. + +```js +class ShareControl { + onAdd(map) { + this._map = map; + this._container = document.createElement('div'); + this._container.className = 'maplibregl-ctrl maplibregl-ctrl-group'; + this._container.style.margin = '0 10px' + + const urlButton = document.createElement('button'); + urlButton.id = 'urlButton' + urlButton.type = 'button'; + urlButton.style.display = 'none' + urlButton.title = 'Copy url to clipboard'; + urlButton.style.borderRadius = "4px"; + urlButton.onclick = () => { + navigator.clipboard.writeText(shareableUrl) + .then(() => { + console.log('URL copied to clipboard: ' + shareableUrl); + const mapContainer = document.getElementById("map"); + const messageContainer = document.createElement("div"); + messageContainer.className = "url-copied-message"; + const icon = document.createElement("i"); + icon.className = "fa-solid fa-link"; + const text = document.createTextNode("URL copied to clipboard."); + messageContainer.appendChild(icon); + messageContainer.appendChild(text); + mapContainer.appendChild(messageContainer); + + // Fade out the message by setting opacity to 0 + setTimeout(() => { + messageContainer.style.opacity = 0; + setTimeout(() => { + mapContainer.removeChild(messageContainer); + }, 500); // Fade out for 500 milliseconds + }, 500); // Displayed solid for 500 milliseconds + }) + .catch(err => { + console.error('Unable to copy URL to clipboard', err); + }); + }; + + const urlIcon = document.createElement('i'); + urlIcon.className = 'fa-solid fa-link'; + urlButton.appendChild(urlIcon); + + const emailButton = this.createShareButton('email', 'fa-solid fa-envelope'); + emailButton.addEventListener('click', () => { + window.open(`mailto:?subject=Check out what has been crossed on my latest adventure!&body=Check out this site ${shareableUrl}`, '_blank'); + }); + const whatsappButton = this.createShareButton('whatsapp', 'fa-brands fa-whatsapp'); + whatsappButton.addEventListener('click', () => { + // https://faq.whatsapp.com/5913398998672934 + if (isRouteDisplayed) { + var whatsappMessage = `Check out the waters I crossed on a recent adventure.. ${shareableUrl}` + } else { var whatsappMessage = `Check out this site for you recent adventures.. ${shareableUrl}` } + let whatsappShareLink = `https://wa.me/?text=${encodeURIComponent(whatsappMessage)}` + window.open(whatsappShareLink, '_blank'); + }); + const facebookButton = this.createShareButton('facebook', 'fa-brands fa-facebook'); + facebookButton.addEventListener('click', () => { + window.open(`https://www.facebook.com/sharer/sharer.php?u=${shareableUrlEncoded}`, '_blank'); + }); + const twitterButton = this.createShareButton('twitter', 'fa-brands fa-twitter'); + // https://developer.twitter.com/en/docs/twitter-for-websites/tweet-button/guides/web-intent + twitterButton.addEventListener('click', () => { + if (isRouteDisplayed) { + var twitterMessage = `Check out the waters I crossed on a recent adventure..` + } else { var twitterMessage = `Check out this site for you recent adventures..` } + window.open(`https://twitter.com/intent/tweet?url=${shareableUrlEncoded}&text=${twitterMessage}`, '_blank'); + }); + this._container.appendChild(urlButton); + this._container.appendChild(emailButton); + this._container.appendChild(whatsappButton); + this._container.appendChild(facebookButton); + this._container.appendChild(twitterButton); + + const shareButton = document.createElement('button'); + shareButton.type = 'button'; + shareButton.title = 'Share'; + shareButton.style.borderRadius = "4px"; + shareButton.onclick = () => { + if (isShareExpanded) { this.minimizeShareControl() } else { this.expandShareControl(); } + }; + + const shareIcon = document.createElement('i'); + shareIcon.className = 'fa-solid fa-share-nodes'; // Assuming you are using FontAwesome for icons + shareButton.appendChild(shareIcon); + + this._container.appendChild(shareButton); + + return this._container; + } + + createShareButton(id, faIcon) { + const button = document.createElement('button'); + button.id = `${id}Button` + button.type = 'button'; + button.style.display = 'none' + button.title = id; + button.style.borderRadius = "4px"; + const icon = document.createElement('i'); + icon.className = faIcon; + button.appendChild(icon); + return button; + } + + minimizeShareControl() { + const urlButton = document.getElementById("urlButton"); + urlButton.style.display = 'none'; + const emailButton = document.getElementById("emailButton"); + emailButton.style.display = 'none'; + const whatsappButton = document.getElementById("whatsappButton"); + whatsappButton.style.display = 'none'; + const twitterButton = document.getElementById("twitterButton"); + twitterButton.style.display = 'none'; + const facebookButton = document.getElementById("facebookButton"); + facebookButton.style.display = 'none'; + isShareExpanded = false + } + + expandShareControl() { + const urlButton = document.getElementById("urlButton"); + urlButton.style.display = 'block'; + const emailButton = document.getElementById("emailButton"); + emailButton.style.display = 'block'; + const whatsappButton = document.getElementById("whatsappButton"); + whatsappButton.style.display = 'block'; + const twitterButton = document.getElementById("twitterButton"); + twitterButton.style.display = 'block'; + const facebookButton = document.getElementById("facebookButton"); + facebookButton.style.display = 'block'; + isShareExpanded = true + } + + onRemove() { + this._container.parentNode.removeChild(this._container); + this._map = undefined; + } + } +``` + +Going deeper + +### ✨ Route shared with magic link ✨ + +I want to avoid adding a backend and having to deal with user data storage. So this must be done on the client side. The site so far is stateless, so the problem is simply how to share the route data from one user to another. + +One way is to enocde it and add it to the url parameters. The polyline format is a perfect choice for this for this, it enocodes a line, or more specifically a series of coordinates into a single string + +Let have a look at how the walk from Brandenburg Tor to the Reichstag would be encoded. +```json +{ + "type": "LineString", + "coordinates": [ + [ + 13.37749, + 52.51626 + ], + [ + 13.3772, + 52.51624 + ], + [ + 13.37703, + 52.5164 + ], + [ + 13.37651, + 52.51638 + ], + [ + 13.37674, + 52.51664 + ], + [ + 13.37442, + 52.51769 + ] + ] +} +``` + +```text +sap_IixspABx@_@`@BfBs@m@qEnM +``` + + +TODO: add codesandbox to let the reader play around. + +That looks much better. This is almost good enough to be used in a url that someone could share through an app. To improve things a little I will avoid them pesky characters `backticks` that mess up the url displaying in whatsapp. + +Anytime a route is processed I update the `shareableUrl` variable with a `route` parameter equal to the encoded polyline. + +```js +shareableUrl = `https://kreuzungen.world/index.html?route=${encodeURIComponent(polyline.fromGeoJSON(displayedRouteGeoJSON))}` +``` +Then +```js +if (urlParams.has('route')) { + var route = urlParams.get('route') + coordinates = polyline.decode(route); + geojson = polyline.toGeoJSON(route) + geojson.properties = { "name": "✨ Route shared with magic link ✨"}; + // Ensure the map style is loaded before processing the route. + if (mapInstance.isStyleLoaded()) { + processGeojson(geojson); + } else { + mapInstance.once('style.load', () => { + processGeojson(geojson); + }); + } +} +``` + +### But how long can a url be? + + + +--- + +#### Diving deeper with the style color + +https://maplibre.org/maputnik/ + + + + ![Screen shot of river geometry intersection](image.png) + is the polygon for the osm water body + is the linestring for the osm waterway way + + +### Speeding things up. + +I want this app to be as fast as possible. Everything feels better when there is a instant reaction, and in the modern "short-attention-span" age, it's kind of a given that things should work quickly and if not people assume something is broken. + +#### Avoid slow requests + +Normally, the Umami analytics script is served from a `/script.js` endpoint on the server that where Umami is runnning. In my case this server is automatically stopped when not in use (for money reasons) and takes more then a jiffy to start up. + +This is suboptimal, and because I am cheap, the start up time of the server is not something I change. I can however change the way the script is loaded. Instead of the client loading the script from the remote server, I can add the script to the built assets and rely on github pages to serve it up fast and efficiently. + + + +#### Async all the things + +There are some things that are "nice to have" but in no way essential for the so called happy path of the app. These should be done in the background and not block the essential parts of the app from working. + + +[`async`](https://developer.mozilla.org/en-US/docs/Glossary/Asynchronous#in_software_design) is a way of doing this, instructing the browser to run a task in the background and not block the main thread. This is a great way to keep the app feeling responsive and fast. + +The umami analytics script is a good example of something that is by no means essential for the app to work but nice to have for some "vanity metrics" telling me how many people are using the app and where they are located in the world. + +It should be loaded in the background and not block the user from interacting with the app. + +```html + + +``` + +For example, when the user uploads a route, the app should display the route on the map as soon as possible, and then fetch the river data in the background. This way the user can start interacting with the map and see the route while the rivers are being fetched. + +### + +#### First things first, lets profile + +TODO: link to profiling proj + +## Build it and they will come, right? + +Ok so the app is built, and it's working. The next step is to share it with the world, and ensure that people can find it when they are looking for it. + +### Memorable domain name + +I already settled on the name Kreuzungen. It's a German word that means "crossings" or "intersections". It's subjectively short, easy to remember, and it's meaning is used in the context of the app. + +To no surprise, the domain kreuzungen.com was already taken and a short call to the owner assured me that it was not up for sale, ces't la internet. + +I looked at what other top level domains were not taken, and although there was not `.en` top level domain so that I could do the cool thing where the fully qualified domain is the world (like `bit.ly`, `redd.it`) I saw that **`krezungen.world`** was available. I liked the way it sounded, so I purchased it. [It should be cool for a long time](https://www.w3.org/Provider/Style/URI). + +### DNS + +Setting up github pages with my new shiny domain was pretty easy. I followed the [docs](https://docs.github.com/en/pages/configuring-a-custom-domain-for-your-github-pages-site) and configured my DNS with a txt record to verify I owned the domain, and some A records pointing to github servers. + +After waiting a few mins, I checked that the records had propagated through the network. + +```bash +➜ ~ dig kreuzungen.world +noall +answer -t A +kreuzungen.world. 195 IN A 185.199.109.153 +kreuzungen.world. 195 IN A 185.199.110.153 +kreuzungen.world. 195 IN A 185.199.108.153 +kreuzungen.world. 195 IN A 185.199.111.153 +``` + +Then I updated the github repo with the custom domain and all seemed good. + +![Github notification saying "Your site is live at https://kreuzungen.world/"](/media/kreuzungen/github_pages_deploy.png) + +A last sanity check with the browser showed that [https://kreuzungen.world](https://kreuzungen.world) was live! + +![https://kreuzungen.world loading for the first time](/media/kreuzungen/loadingkreuzungen.gif) + +I set up the www. subdomain to redirect to the "naked" apex domain, because [both is better](https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Choosing_between_www_and_non-www_URLs#make_your_page_work_for_both). + +I also made records to point the subdomains `auth.kreuzungen.world` and `stats.kreuzungen.world` to the fly.io servers running the backend and analytics services. This is for cosmetic reasons has no technical advantage (subdomains are still different domains, so CORS still shows up with a arguable performance cost). I avoid the user seeing any requests to `https://random_scary_name.fly.dev/`, and that makes me happy. + + + +### SEO + +I am no SEO expert, and everything I implement was based on a few google searches, if any SEO pro's are reading and have some tips, I would love to hear them. + +#### Google + +It is 2024 and people google things, that is a fact of life and if I want people to be able to find my site, I need to make sure it is displayed on google. Therefore I need to get google to crawl and index it! + +{{< admonition type=tip title="Google 101" >}} +Generally, google finds all website and will automatically index them, but this can take time + +There are a few things you can do to speed up the process. +{{< /admonition >}} + +#### Crawl first before you walk + +I added the site to the [google search console](https://search.google.com/search-console/about) and verified that I was the owner of the domain by adding a TXT record to my DNS. + +![Kreuzungen.world listing on Google!](/media/kreuzungen/google.png) + +#### Description + +TODO: talk about how you imrpoved the description/snippet [https://developers.google.com/search/docs/appearance/snippet](https://developers.google.com/search/docs/appearance/snippet) with SEO magic. + +## Getting hooked into that webhook stuff + +I was inspired by what I have seen from [https://wandrer.earth](https://wandrer.earth), [https://summitbag.com](https://summitbag.com) and [https://whatismyskc.com](https://whatismyskc.com) on my friends activities. Theses apps enrich user activities by pushing additional information back to Strava servers, such that the greater Strava community to see it. + +I wanted to have a similar feature in my app, a automagical feature that updates the description of an activity with a nice message including the name and number of waterways crossed, and a link to [https://kreuzungen.world](https://kreuzungen.world). + +{{< admonition type=tip title="Update message" open=true >}} + + +{{% center %}} +Crossed 5 waterways 🏞️ Nile | Amazon River | Mississippi River | Danube River | Ganges | River Thames 🌐 https://kreuzungen.world 🗺️ + +TODO: add a screenshot of it in action + +_*if any design people are reading this, I would love some help to improve this message_ + +{{% /center %}} +{{< /admonition >}} + +To accomplish this, I utilized the [Strava Webhook Events API](https://developers.strava.com/docs/webhooks/). I created a subscription and wrote a little server to listen for events from the subscription, process the new activites and update the activity descriptions on Strava in real-time. + +This approach eliminates the need for users to manually check activities on [https://kreuzungen.world](https://kreuzungen.world). After a one-time authorization of Kreuzungen with the "Upload your activities from Kreuzungen to Strava" scope set, users should receive automatic and consistent updates about the waterways on their Strava profile, for life. + +The enriched activity descriptions are then visible to all on the Strava platform, potentially inspiring others to explore Kreuzungen and join a new community of river lovers. More people using the kreuzungen.world with strava means more people will see the app, meaning more people will use the app, and so on. Viva la Kreuzungen! + +### Refactor + +Krezungen was until now a webapp written in javascript and running in the browser. I wanted to use the same logic used to process routes in the browser, but I wanted this to run on a lightweight server somewhere that strava could talk to it. + +One option I had was to translate the geometry processing code, into my fave language python. + +Another way would be to use node to run some of the same code that is used in the frontend. This would be a good way to avoid reimplementing the wheel (or most parts atleast), and keeps a single, nodivergent codebases to work with in the future. + +I would need to refactor the code to run it with node. I decided to go with this option, and refactor the code and use TypeScript and use webpack to bundle up the assets needed for the webapp. + +The structure would look a little like this: + +| File | Description | +|-----------|--------------------------------------------| +| geo.ts | Contains the geometry processing code | +| strava.ts | Contains the logic to auth with Strava | +| main.ts | Contains the code to be bundled and injected into the webapp | +| app.ts | Contains the Express server and webhook subscriptions | + +### Setting up a webhook subscription + +I needed to set up a webhook subscription with Strava to receive events when a user uploads a new activity. I followed the [Strava Webhook Events API](https://developers.strava.com/docs/webhooks/) documentation. + +This is a two step part which essentaily involves: + +1. setting up a server to listen which will respond correctly to a validation request from Strava +2. making a POST request to the Strava API to create the subscription. + +First I created `app.ts` to handle the webhook subscription callback validation flow. + +```typescript +// route for '/webhook' to verify the webhook subscription with Strava +app.get("/webhook", (req, res) => { + const VERIFY_TOKEN = "STRAVA"; + + let mode = req.query["hub.mode"]; + let token = req.query["hub.verify_token"]; + let challenge = req.query["hub.challenge"]; + + if (mode && token) { + if (mode === "subscribe" && token === VERIFY_TOKEN) { + console.log("WEBHOOK_VERIFIED"); + res.json({ "hub.challenge": challenge }); + } else { + res.sendStatus(403); + } + } +}); +``` + +I deploy this to fly.io and register the webhook with Strava with a POST request to the Strava API. + +```bash +curl -X POST https://api.strava.com/api/v3/push_subscriptions \ + -H "Authorization + -H "Content-Type: application/json" \ + -d '{"client_id": 12345, "client_secret": "supersecret", "callback_url": "https://yourdomain.com/webhook", "verify_token": "STRAVA"}' +``` + +## Squishing bugs + +During refactoring I found a bug in the code that was causing the random rivers to creep into the results. After some digging and test writing, it looked like extra waterways were being selected as intersecting with the route when they had a self intersection. + +I looked into the turf [booleanIntersects](https://www.npmjs.com/package/@turf/boolean-intersects) implementation, and it turns out that one of the underlying functions lineIntercept() has a `ignoreSelfIntersections` vaiable that can be set. I fixed this locally and the tests passed. + +```bash +kreuzungen-py3.11➜ site git:(new) ✗ npm test + +> test +> jest --verbose + + PASS src/geo.test.ts (5.71 s) + intersectingFeatures + ✓ should return a FeatureCollection of 2 intersecting kanals in nk (3 ms) + ✓ should return a FeatureCollection of 1 intersecting features including Grünerbach and excluding Mandlerbach (15 ms) + ✓ should return a empty FeatureCollection of intersecting features excluding Mandlerbach (4 ms) + ✓ should return a FeatureCollection with a single feature Grünerbach (6 ms) + +Test Suites: 1 passed, 1 total +Tests: 4 passed, 4 total +Snapshots: 0 total +Time: 5.844 s +Ran all test suites. +``` + +I want to reuse the geometry processing code that I wrote for the frontend to process the strava activities. I could have copied the code, but that would have been a bad idea, because if I ever wanted to change the way the geometry is processed, I would have to change it in two places. I decided to refactor the code into a separate module that could be imported into the backend. + +#### webpack + +TODO: explain a bundler and why it is good to use one here + +```js +const HtmlWebpackPlugin = require('html-webpack-plugin'); +const path = require('path'); + + +module.exports = { + mode: 'development', + devtool: 'inline-source-map', + devServer: { + static: './dist', + }, + optimization: { + runtimeChunk: 'single', + }, + entry: { main: './src/main.ts' }, + module: { + rules: [ + { + test: /\.ts?$/, + use: 'ts-loader', + exclude: [/node_modules/, /\.test\.ts$/], + }, + { + test: /\.(png|jpe?g|gif|svg)$/i, + type: 'asset/resource' + }, + { + test: /\.css$/i, + use: ['style-loader', 'css-loader'], + exclude: /input\.css$/, + }, + { + test: /\.md$/, + use: [ + { + loader: 'html-loader', + options: { + esModule: false, + }, + }, + { + loader: 'markdown-loader', + }, + ], + }, + ], + }, + resolve: { + extensions: ['.tsx', '.ts', '.js'], + }, + node: { global: true }, + plugins: [ + new HtmlWebpackPlugin({ + template: './src/index.html', + inject: true, + }), + ], + output: { + filename: '[name].bundle.js', + path: path.resolve(__dirname, 'dist'), + }, +}; +``` + +tsconfig.json + +```json +{ + "compilerOptions": { + "target": "es2016", + "module": "commonjs", + "outDir": "./dist", + "esModuleInterop": true, + "forceConsistentCasingInFileNames": true, + "strict": false, + } + , "files": [ + "./src/app.ts", +] +} + +package.json + +```json +{ + "devDependencies": { + "@types/jest": "^29.5.12", + "@types/lodash": "^4.17.0", + "@types/mapbox__polyline": "^1.0.5", + "@types/marked": "^6.0.0", + "@types/umami": "^0.1.5", + "css-loader": "^7.1.1", + "html-loader": "^5.0.0", + "html-webpack-plugin": "^5.6.0", + "jest": "^29.7.0", + "markdown-loader": "^8.0.0", + "marked": "^12.0.2", + "style-loader": "^4.0.0", + "tailwindcss": "^3.4.3", + "ts-jest": "^29.1.2", + "ts-loader": "^9.5.1", + "typescript": "^5.4.5", + "webpack": "^5.91.0", + "webpack-cli": "^5.1.4", + "webpack-dev-server": "^5.0.4" + }, + "scripts": { + "test": "jest --verbose", + "build": "webpack", + "serve": "webpack serve --open", + "compile": "npx tsc", + "start": "node dist/app.js" + }, + "dependencies": { + "@fortawesome/fontawesome-svg-core": "^6.5.2", + "@fortawesome/free-brands-svg-icons": "^6.5.2", + "@fortawesome/free-solid-svg-icons": "^6.5.2", + "@mapbox/polyline": "^1.2.1", + "@mapbox/togeojson": "^0.16.2", + "@turf/boolean-point-in-polygon": "^7.0.0-alpha.115", + "@turf/helpers": "^7.0.0-alpha.115", + "@turf/line-intersect": "^7.0.0-alpha.115", + "@turf/meta": "^7.0.0-alpha.115", + "@turf/polygon-to-line": "^7.0.0-alpha.115", + "@turf/turf": "^7.0.0-alpha.114", + "@umami/node": "^0.2.0", + "body-parser": "^1.20.2", + "cross-fetch": "^4.0.0", + "express": "^4.19.2", + "font-awesome": "^4.7.0", + "fontawesome-free": "^1.0.4", + "geojson": "^0.5.0", + "lodash": "^4.17.21", + "maplibre-gl": "^4.1.2", + "osmtogeojson": "^3.0.0-beta.5", + "redis": "^4.6.13" + } +} +``` + +### Sprinkle Redis in the Mix + +I also needed to bite the bullet and admit it was time to set up a perisitant storage. + +Up until now, I avoided the need of any storage. However if I want to process strava activities on behalf of users, I would need to re-authenticate with strava without the user . + +I added a fly.io redis instance to the backend to store the strava access tokens and refresh tokens. This makes me a full stack developer now, right? + +I set env vars for the redis url and password on fly.io and added the following code to the backend to store the tokens. + +### Express server and webhook subscriptions + +I wanted to have a lightweight backend, and I decided to use express.js to serve the frontend and handle the strava webhooks. + +I created a file `src/app.ts` and added the following code to anwser the strava webhook verification and process the incoming events. + + +```typescript +import express from "express"; +import bodyParser from "body-parser"; +import { createClient } from "redis"; +import { + getStravaAccessTokenRedis, + getStravaActivity, + updateStravaActivityDescription, +} from "./strava"; +import { calculateIntersectingWaterwaysPolyline, createWaterwaysMessage } from "./geo"; + +const app = express().use(bodyParser.json()); +const redisClient = createClient({ url: process.env.REDIS_URL }); +redisClient.on("error", (error) => { + console.error(`Redis client error:`, error); +}); +main(); + +// The main asynchronous function +async function main() { + try { + await redisClient.connect(); + app.listen(process.env.PORT || 80, () => + console.log("webhook is listening") + ); + } catch (error) { + console.error("Error connecting to Redis:", error); + } +} + +// route for '/webhook' to verify the webhook subscription with Strava +app.get("/webhook", (req, res) => { + const VERIFY_TOKEN = "STRAVA"; + + let mode = req.query["hub.mode"]; + let token = req.query["hub.verify_token"]; + let challenge = req.query["hub.challenge"]; + + if (mode && token) { + if (mode === "subscribe" && token === VERIFY_TOKEN) { + console.log("WEBHOOK_VERIFIED"); + res.json({ "hub.challenge": challenge }); + } else { + res.sendStatus(403); + } + } +}); + +// route for '/webhook' to receive and process incoming events +// must acknowledge each new event with a status code of 200 OK within two seconds. +// Event pushes are retried (up to a total of three attempts) if a 200 is not returned. +// If your application needs to do more processing of the received information, it should do so asynchronously. +app.post("/webhook", async (req, res) => { + // return a aknoledgment and process the event asynchronously + res.status(200).send("EVENT_RECEIVED"); + const event = req.body; + if (event.aspect_type === "create" && event.object_type === "activity") { + processAndUpdateStrava(event.owner_id, event.object_id); + } + else if (event.aspect_type === "update" && event.object_type === "athlete" && event.updates && event.updates.authorized === "false") { + deleteUser(event.owner_id) + } +}); + +async function processAndUpdateStrava(owner_id, activity_id,) { + try { + const owner_access_token = await getStravaAccessTokenRedis( + owner_id, + redisClient + ); + if (!owner_access_token) { + console.error("No access token found for user_id: " + owner_id); + return; + } + const activityData = await getStravaActivity( + activity_id, + owner_access_token + ); + + if (!activityData.map || !activityData.map.summary_polyline) { + console.error( + "Activity does not have a summary polyline:", + activityData + ); + return; + } + + const intersectingWaterways = await calculateIntersectingWaterwaysPolyline( + activityData.map.summary_polyline + ); + + if (intersectingWaterways.features.length === 0) { + console.log("No intersecting waterways found"); + return; + } + + const waterwaysMessage = createWaterwaysMessage(intersectingWaterways); + + // update the activity description with the waterways message if there are waterways + const success = await updateStravaActivityDescription( + activity_id, + owner_access_token, + waterwaysMessage + ); + if (success) { + console.log(`Updated activity https://www.strava.com/activities/${activity_id} with ${waterwaysMessage}`) + } else { + console.error(`Failed to update activity description for activity_id: ${activity_id}`); + } + } catch (error) { + console.error("Error updating activity description", error); + } +} + +async function deleteUser(owner_id) { + try { + await redisClient.del(owner_id.toString()); + console.log(`Deleted access token for user_id: ${owner_id}`); + } catch (error) { + console.error("Error deleting access token", error); + } +} +``` + + +```python + +``` +TODO: copy code +``` + +```bash +TODO: copy code +``` + +### Strava API webhooks + +TODO: add video + +TODO: add section on manual update button. + +## Adding a faq section + +I wrote up some common questions that I thought people might have, and added them to the site. I kept this simple and used markdown to write the questions and answers, and then used the html-loader and markdown-loader to load the content into the site. + +I then added a map control using a question mark icon that would toggle the faq section on and off. + +## Improving the UI + +[UI/UX Best Practices for Designing Amazing Web Apps ](https://youtu.be/OSfSDdl-QmM) +https://youtu.be/OSfSDdl-QmM?t=955 + +[https://www.mapuipatterns.com/](https://www.mapuipatterns.com/) + + +## Tracking + +TODO: write about wanting to have privacy respecting analytics, and how I self hosted umani and wrote some basic events to get a understanding of how the app is used and how this could be useful in the future to improve the user experiance. + +I wanted some + +TODO: Show graph + +### Future imrpovements + +Use this lib for having gpx and osm stuff straight onto the map. +[https://github.com/jimmyrocks/maplibre-gl-vector-text-protocol](https://github.com/jimmyrocks/maplibre-gl-vector-text-protocol) + +### Hindsight + +Typescript is great. I should have used it from the start. It would have saved me a lot of time debugging and writing tests, and will probably save me time in the future. + +If I started again I would start out using a with a framework that deals with state. In the end the ui became more complex, and I had to use a lot of global variables to keep track of the state of the app. Using a framework would outsource + +When I implemented the share controller, I got a bit caried away and added twitter/facebook/whatsapp share buttons. I should have just kept it simple and only added the copy to clipboard button. The other share buttons are not really needed, and apparently people do not share on social media, and if they do they copy the url and paste it. Removing the buttons reduced some code and made the ui cleaner. + +I thought it was a good idea to add the sharing of routes by encoding them into a polyline and adding the string to a url parameter. This has arguable benefits, as the route can be shared without the need for a backend to store the precious user data. + +However the reality was that this doesn't work, because of the way messaging apps like whatsapp handle super long urls. Only some of the url is displayed, and the user must explictly expand/uncollapse it to see the full url. If they press on the shortened url, it will open in the browser but only show the trucated route. This is a confusing/buggy user experience, and while I really like providing users the option of easily sharing a route without storing the data on kreuzungen servers, most people prefer the convenience of a simple share button and a freindly url like . + +### Conclusion + +Like the ride, I set out on this project, not knowing how I would get all the way to the end, but with a naive courage that I could do it. In life it is always good to try things and just start out. I learnt a lot about mapping technologies. I also learnt about the turf library and how to use it to process geojson data. I also learnt about the OpenStreetMap data model, the overpass api and how to use it to get data about waterways, and I learnt about the strava api and webhooks. Along the way I learnt about typescript and how to use it to write more robust code and how to use webpack to bundle up the code. + +Looking back I am quite chuffed with how far I have come and what I have achieved. + +It is really cool seeing the app being used in distant place like the US, Australia, Serbia or Norway. And it was amazing to see it pop up on my strava feed, from a freind of mine that had no idea I made this. + +TODO: insert screenshot with the comment + +I am happy that people are enjoying this app, and I hope it inspires people to get out and explore the waterways around them. + +TODO: add image of peoples strava usage around the world. + +There is a magical thing about the internet, building something from open data and seeing it used by people all over the world. + +I am happy with the results, I anwserd the question I set out to answer, I have a working app that I can share with the world. I looking forward to seeing if the app takes of and reaches a level of viralbility to take off. + +One of the reasons I got into programming in the first place was this romantic idea that the time that I invested once codiefied could be multiplied infinitely. Now I have finished, I am sure that this will blow up. + +#### Known issues + +When asking OSM for ways and joining together ways with the same name, there might be some common name used for them both (), meaning there will be a . + +Sometimes the name fo the relation willbe different from the ways resulting in two unique objects. + +Zwift is a virtual cycling app for people who ride indoors infront of a screen. + +![Zwift website]() + +It is often connected with Strava and the routes use real world geographies for the virtual routes[^zwift]. Some of these routes intersect with real life waterways, like the central park route intersecting with [The Loch](https://www.openstreetmap.org/node/5214031647#map=15/40.7952/-73.9578). Not sure if that's a bug or a feature but yeah... This is something I didn't prepare for and leaves me feeling weird. (On a side note, the Zwift tech community unexpectaly goes in deep [https://zwiftmap.com/watopia](https://zwiftmap.com/watopia), [https://zwiftinsider.com/route/park-perimeter-reverse/](https://zwiftinsider.com/route/park-perimeter-reverse/)) + +![](image-1.png) + +### TODO: + +[ ] - add diagrams https://hugodoit.pages.dev/create-diagrams/#complicated +[ ] - add contents on the right hand side + + + + + + + + +## Other interesting shit + +- Cool web app that shows the watershed from any point clicked on the map! +[https://mghydro.com/watersheds/](https://mghydro.com/watersheds/) +- Global river runner, api for calculating the path any raindrop will run out to the ocean [https://ksonda.github.io/global-river-runner/](https://ksonda.github.io/global-river-runner/) +- Visulaise the path of a rain drop falling to the ocean [https://river-runner-global.samlearner.com/](https://river-runner-global.samlearner.com/) +- web app showing connected waterway networks [waterwaymap.org](waterwaymap.org) +- OSM forum discussion walking about if waterways should be mapped across lakes ect.. [https://community.openstreetmap.org/t/should-river-lines-be-mapped-through-lakes-estuaries-gulfs-and-other-large-water-bodies/104438]https://community.openstreetmap.org/t/should-river-lines-be-mapped-through-lakes-estuaries-gulfs-and-other-large-water-bodies/104438() + +## Footnotes + +[^dicke-marie]: [https://nationalerbe-baeume.de/project/dicke-marie-berlin-tegel/](https://nationalerbe-baeume.de/project/dicke-marie-berlin-tegel/) +[^river-wiki]: [https://en.wikipedia.org/wiki/River](https://en.wikipedia.org/wiki/River) +[^osm-services]: [https://wiki.openstreetmap.org/wiki/List_of_OSM-based_services](https://wiki.openstreetmap.org/wiki/List_of_OSM-based_services) +[^start-stop]: [https://en.wikipedia.org/wiki/Start-stop_system](https://en.wikipedia.org/wiki/Start-stop_system) +[^zwift]: [https://www.bikeforums.net/indoor-stationary-cycling-forum/1286793-question-zwift-users.html](https://www.bikeforums.net/indoor-stationary-cycling-forum/1286793-question-zwift-users.html) \ No newline at end of file diff --git a/content/posts/trains.md b/content/posts/trains.md new file mode 100644 index 0000000..c0884f7 --- /dev/null +++ b/content/posts/trains.md @@ -0,0 +1,791 @@ +--- +title: "How I took trains (not planes) and reached the Pyrenees 💡" +subtitle: "" +date: 2024-08-14T17:04:23+02:00 +lastmod: 2024-08-14T17:04:23+02:00 +authors: [] +description: "" + +tags: [] +categories: [] +series: [] + +hiddenFromHomePage: false +hiddenFromSearch: false + +featuredImage: "" +featuredImagePreview: "" + +toc: + enable: true +math: + enable: false +lightgallery: true +license: "" +--- + + + +## intro 🚂 + +I was invited to join some friends on a mountain bike tour in the Pyrenees. I was ready, as was my bike. The only thing left to solve was how to get there. + +TODO: insert a picture of my bike on a globe, and a lightbulb idea to take regional trains. maybe a google map screenshot. + +This is the story of how I took regional times and went through 5 international boarders along the way. (Austria, Italy, Monaco, France 🇦🇹->🇮🇹->🇫🇷->🇲🇨->🇫🇷). + +I hope it helps anyone who is thinking about taking a similar journey. + +## The route 🗺️ + +TODO: insert a maplibre-gl map with the route from the journey data. + +## Planning 📅 + +So first things first, you need to plan your route and know which way you are going. + +I must confess I sinned and used the mighty google maps to help planning and decide which route to take. It actually has great coverage across europe and everyone knows that google maps kills it in terms of a easy and quick user experience. + +{{< image src="/media/trains/google_maps.png" alt="Screenshot of Google Maps" >}} + +Bear in mind that your mileage may vary, depending on where you want to travel in the world and there are other tools out there. + +{{}} +I hear very good things from my friends about [www.trainline.eu](www.trainline.eu). + +There is also a couple projects out there which are handy when planning such a route as mine. [https://www.bahn.guru/](https://www.bahn.guru/) is a great tool for planning train routes in Europe. It is a open source project and has a great community behind it. I also used [https://www.chronotrains.com/en](https://www.chronotrains.com/en) to check the train times and prices. +{{}} + +I needed to travel from Innsbruck in Austria to la tour de carol in France, which meant I would need to traverse around the alps and then south to the pyraness. I entered the start and end points into google maps and it gave me a few options including routes via Zurich, Paris and Milan. + +I considered the different options and decided to take regional trains for this journey through italy and the french coast. It would of actually been quicker to take the high speed train via zurich and paris, but not significantly. Plus I had the privilege of free time on hand and the luxury of deciding where I wanted to be with having to be at a specific place at a specific time. This meant that I could split up the journey and enjoy some of the beautiful places along the way. Booyah! + +Taking the regional trains was a easy choice, as although slower, they have more space for bikes. I had some beautiful views of the Alps, the Italtian Rivera and Côte d'Azur. It also meant I went through Monaco, and could weave between the super expensive sports cars of the rich people stuck in traffic on my bike, living the best life. + +## Bike reservation 🚳 + +The big question: Can you take your bike on the train and do you need to reserve a spot for your bike? The answer is, it depends. + +Most european trains have spaces for bikes. A reservation is usually mandatory for high speed trains. In the summer months, there reservations are often sold out, so it is best to book in advance. + +Some high speed trains do allow fully assembled bikes, but this and not having a reservation can be worked around with a rinko bike bag set-up. + +Regional trains are the best for bike travel, almost always have space for bikes and you don't need a reservation. + +There are some good resources out there which list the reservation conditions for each train company. + +- [https://help.raileurope.com/article/41750-taking-bikes-on-the-train](https://help.raileurope.com/article/41750-taking-bikes-on-the-train) + +Otherwise, each of the train providers has a webpage with all the infomation you need. + +- [🇩🇪 Deutsche Bahn](https://www.bahn.de/angebot/zusatzticket/checkliste-fahrradmitnahme) +- [🇦🇹 Öbb](https://www.oebb.at/en/reiseplanung-services/im-zug/fahrradmitnahme) +- [🇫🇷 SNCF](https://www.sncf-voyageurs.com/en/travel-with-us/train-and-bike/bike-on-board/) +- [🇮🇹 Trenitalia](https://www.trenitalia.com/en/services/travelling_with_yourbike.html) +- [🇪🇸 Renfe](https://www.renfe.com/es/en/travel/informacion-util/luggage/bicycles-and-kick-scooters-_non-electric_) +- [🇳🇱 NS](https://www.ns.nl/en/travel-information/bikes-on-the-train.html) + +{{}} + +When using a train carrier website, you can usually find the filter the connections to only show those that allow bikes. This makes planning with a bike easy peazy lemon squeezey 🍋. + +{{}} + +### My reservation learnings 🚲 + +Everyones trip will be different depending on the region they are travelling, but here is what I experienced on my journey. + +#### Öbb 🇦🇹 + +The first trip I took involved a s-bahn from Innsbruck up to the Brenner pass. This is a historical place known for. And the train winds through the valley offering beautiful views all the way. + +You don't have to make any reservation, but need to buy a bike ticket for TODO:. There is ample space on the modern trains to carry bikes. + +TODO: insert picture of the sbahn. + +#### Trenitalia 🇮🇹 + +There are multiple train operators in Italy. Trenitalia is the national company, but there is also Trenord, TODO: add the others. Possibilities to take the bike on board vary depending on which company owns the train. + +I took the regional trains through the northen part of the country. + +Between Brenner and Verona, I + +Then I needed to go via Milano, to Genova and then through Liguria along the Italian Rivera to the french Boarder. + +A bike ticket costs a extra 3.50. + +It's possible to take a french train from Ventimiglia, the last town in Italy to Nice in France, and you can take your bike with you. I decided to ride that section because I wanted to ride across Monaco. + +#### SNCF 🇫🇷 + +TODO: fact check this + +France has a very good high speed network, with very rapid trains (>250km/h) connecting the north to the south. However the bike spaces on these super fast trains are few or don't exist at all. I witnessed the Rinko bike set up was a big thing in France, probably as a result. + +It is possible to take your bike on every regional train, which have good facilities for bikes. Sometimes you need to make a reservation, and sometimes not. This varies region to region. + +It is free to take your bike on TER regional trains in the Occitanie region, but you should make a online bike reservation to avoid any problems if the train is packed :wink:. If you don't have one and the train is full you might be asked to leave the train. + +Booking a reservation in this region can only be done online using (acces-velo-serein-liotrain.com)[acces-velo-serein-liotrain.com], its a easy process. TODO: check thisacces-velo-serein-liotrain.com + +TODO: insert a gif of the booking process... + +The French Cote da azure region similarly requires a reservation to take a bike on regional trains, but only some train lines. This is done using TODO:nsert website. However, the line I was on (marseille -> Nice) didn't require one. + +## The tour 🚵 + +The tour was a good one, lots of big mountain fun with the gang! + +TODO: add video + +I loved it. Here is a map showing the route that we took and some interesting features we passed. + +By the way, lots of the time we actually where hike-and-biking. This is a insane passion we share for pushing a bike up a hill for hours just to ride back down again (and sometime down when the rocks are too gnar to ride). + +TODO: add the map of the route including pics at the right places! add river crossings from kreuzungen and calculate board crossing and other monuments passed. + +🇫🇷 +🇪🇸 +🇦🇩 + +A interesting geographical thing along the route was the enclave/exclave of Livia. this little town + hill is Spanish (debatably Catalan). Basically when Spain and France made the deal to create the modern day boarded, it included a term about all the towns on the northern side becoming French. Livia was in a loop-hole situation and said that is was not a town, so the rules didn't apply and some how that held up. History is nuts! + +## Route 🗺️ + +- Innsbruck -> Brenner (S bahn) €10.10 + €2.60 for the bike +- Brenner -> Verona (Trente Italia Regionale 204 bike 4*) €26.60 + €3.50 Train no. 3845 + - I actually had some time to wait, so took the opertunity to cruise down hill on my faveriot cycle path to the next station. + - The most bike room ever, like 2 full carts (check how many). + - Over Night in Verona @ castel Camping + - wake up before the sunrise, pack the tent and ride through the romantic city of verona. +- Verona -> Milan (1:20) + - Brought the ticket online + - TODO: insert a pic. +- Milan -> Ventimiglia (4:00) +- Ventimiglia -> Nice (Bike riding. Went through the whole of Monaco!) + I arrived around for lunch in Ventimiglia and then hit the road. It was stunning! I stuck to the coast and was quickly speeding through a tunnel to a signpost signify the french boarder. I love a good schegen crossing on my bike +- Nice -> Marseille St. Charles (SNCF regional TER 20:25 -> 23:03 €39.80 free bike bike 5* power wifi) + - Amazing useage of web maps on the "connect to wifi" landing page ![Screen shot of the SNCF map https://wifi.sncf/en/journey](image.png) +- Marseille -> Narbonne +- Narbone -> + +```json +[ + { + id: 1, + start_station: Innsbruck Hauptbanhof + end_station: Brenner + start_time: "13:41" + end_time: "" + delay_mins: 0 + cost_euros: + train_company: ÖBB + train_type: S-bahn (city -> suburbs) + wifi: No + toilets: yes + bike_space_rating: 4 + geojson: {} + }, +] +``` + +## Return route + +tour de carol -> Toulouse Matabieun TER871470 7:29 €12 +Toulouse -> Narbonne TER86975 €30.00 +Narbonne -> Marseille TER876542 14:31 -> 17:48 €45.20 +Marseille -> Nice TER17493 18:57 -> 21:38 €39.80 +Nice -> Ventimiglia TER86097 22:19 -> 21:11 €9.20 + +I stopped in Menton, the last town in France for the night and stayed at a beautiful campsite up a hill, overlooking the french rivera. + +I set no alarm and the next day I woke up to beautiful views. I rode down through the city, and cruised along the coast, through the boarder to italy. It is a incredibly beautiful part of the world, with the coast and the mountains behind it. + +Arriving in Ventimiglia, I checked out the train situation and learnt that the first train to Milan was full. I could pay a extra €50 to ride anyway, which explicitly includes having no seat, or I could wait 2 hours for the next regional. Having no time pressure I went for the later option, and enjoyed a ride along the coast to San Remo along with a swim to cool me off along route 🏊. + +A big highlight was the old train tunnel which is now a bike only tunnel and has a exhibit which plays homage to Milan San Remo race. The was really cool. Also I was suprised by the train station, a super modernist i-dont-even-know architecture and the platform was 500m into the hill. + +San Remo -> Savona RE12267 12:46 -> 14:25 €34.50 + €3.50 +Savona -> Milano Lambrate RE3035 14:35 -> 17:27 inlcuded with the above +Milano Lambrate -> Verona Porta Nuevo RE2635 17:33 -> 19:17 included with the above + +** There was a issue with the train to milano arriving late and missing my connection to Verona. As a result I took the Trenord to Brescia and then to Verona. + +Milan Lambrate -> Brescia R4 Trenord 17:59 -> 18:53 +Brescia -> Verona R2637 + +I spent another night at my favorite campsite in Verona, and then took the train to Bolzano the next day. + +Verona -> Bolzano Trenteitalia 11:06 -> 13:23 €14.90 + €3.50 + +## Putting the map together + +I wanted to make an interactive map to show the journey, the train routes and the bike rides. With the following features: + +- train stations as Points +- train legs displayed as LineString, the color should represent which country the train is +- bike rides displayed as LineString, the color should represent the bike route +- Hover actions to show information about the journey leg +- extra points of interest along the way + - Campsites + - Pictures + +### Train data + +Along the journey, I noted down the different trains that I took, and included the arrival and destination times, the ticket price, if the train had wifi and rated how good the bike carrying set up was. + +I structured this data as a json with the following schema: + +```json +{ + "type": "array", + "items": { + "type": "object", + "properties": { + "id": { + "type": "integer", + "description": "A unique identifier for each leg of the journey." + }, + "start_station": { + "type": "string", + "description": "The station where the journey leg starts." + }, + "end_station": { + "type": "string", + "description": "The station where the journey leg ends." + }, + "start_time": { + "type": "string", + "format": "time", + "description": "The time the journey leg begins (in HH:MM format)." + }, + "end_time": { + "type": "string", + "format": "time", + "description": "The time the journey leg ends (in HH:MM format)." + }, + "delay_mins": { + "type": "integer", + "description": "Delay in minutes, if any." + }, + "cost_euros": { + "type": ["number", "string"], + "description": "The cost of the journey leg in euros. Can be a number or an empty string if unknown." + }, + "train_company": { + "type": "string", + "description": "The company operating the train." + }, + "train_number": { + "type": ["string", "null"], + "description": "The specific train number for the journey leg. Nullable if not applicable." + }, + "train_type": { + "type": "string", + "description": "The type of train or mode of transportation." + }, + "wifi": { + "type": "string", + "description": "Indicates whether Wi-Fi is available on the train (Yes/No)." + }, + "toilets": { + "type": "string", + "description": "Indicates whether toilets are available on the train (Yes/No)." + }, + "bike_space_rating": { + "type": ["integer", "string"], + "description": "Rating of the bike space on the train. Can be a number or 'N/A' if not applicable." + }, + "geojson": { + "type": "object", + "description": "Placeholder for GeoJSON data for the route (geometry and properties)." + } + }, + "required": [ + "id", + "start_station", + "end_station", + "start_time", + "end_time", + "delay_mins", + "cost_euros", + "train_company", + "train_type", + "wifi", + "toilets", + "bike_space_rating", + "geojson" + ], + "additionalProperties": false + } +} +``` + +The resulting return route looked like this: + +```json +[ + { + "id": 7, + "start_station": "Tour de Carol", + "end_station": "Toulouse Matabieu", + "start_time": "07:29", + "end_time": "", + "delay_mins": 0, + "cost_euros": 12, + "train_company": "SNCF", + "train_number": "TER871470", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": "N/A", + }, + { + "id": 8, + "start_station": "Toulouse", + "end_station": "Narbonne", + "start_time": "", + "end_time": "", + "delay_mins": 0, + "cost_euros": 30, + "train_company": "SNCF", + "train_number": "TER86975", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": "N/A", + }, + { + "id": 9, + "start_station": "Narbonne", + "end_station": "Marseille", + "start_time": "14:31", + "end_time": "17:48", + "delay_mins": 0, + "cost_euros": 45.20, + "train_company": "SNCF", + "train_number": "TER876542", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": "N/A", + }, + { + "id": 10, + "start_station": "Marseille", + "end_station": "Nice", + "start_time": "18:57", + "end_time": "21:38", + "delay_mins": 0, + "cost_euros": 39.80, + "train_company": "SNCF", + "train_number": "TER17493", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": "N/A", + }, + { + "id": 11, + "start_station": "Nice", + "end_station": "Ventimiglia", + "start_time": "22:19", + "end_time": "21:11", + "delay_mins": 0, + "cost_euros": 9.20, + "train_company": "SNCF", + "train_number": "TER86097", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": "N/A", + }, + { + "id": 12, + "start_station": "San Remo", + "end_station": "Savona", + "start_time": "12:46", + "end_time": "14:25", + "delay_mins": 0, + "cost_euros": 34.50, + "train_company": "Trenitalia", + "train_number": "RE12267", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": 3, + }, + { + "id": 13, + "start_station": "Savona", + "end_station": "Milano Lambrate", + "start_time": "14:35", + "end_time": "17:27", + "delay_mins": 0, + "cost_euros": "", + "train_company": "Trenitalia", + "train_number": "RE3035", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": 3, + }, + { + "id": 14, + "start_station": "Milano Lambrate", + "end_station": "Verona Porta Nuova", + "start_time": "17:33", + "end_time": "19:17", + "delay_mins": 0, + "cost_euros": "", + "train_company": "Trenitalia", + "train_number": "RE2635", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": 3, + }, + { + "id": 15, + "start_station": "Milano Lambrate", + "end_station": "Brescia", + "start_time": "17:59", + "end_time": "18:53", + "delay_mins": 0, + "cost_euros": "", + "train_company": "Trenord", + "train_number": "R4", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": 3, + }, + { + "id": 16, + "start_station": "Brescia", + "end_station": "Verona", + "start_time": "", + "end_time": "", + "delay_mins": 0, + "cost_euros": "", + "train_company": "Trenitalia", + "train_number": "R2637", + "train_type": "Regional", + "wifi": "No", + "toilets": "Yes", + "bike_space_rating": 3, + } +] +``` + +The next step was adding the geospatial data of the train route. + +To get the geometric data of the train journeys I would have to consult the internet. + +Like most geospatial problems, this can solved in numerous differnt ways, and a good knowledge of the tools available is key to finding the best solution for the problem at hand. I wish I could import knowledge of all tools out there into my brain, but I can't. So I have to rely on my experience and searching the internet to help me out. + +The gtfs data format is a great resource for train data. It stands for General Transit Feed Specification and is a format for public transportation schedules and associated geographic information. It is used by many public transportation agencies to share their data with developers and other interested parties. This is what is used by google maps to show you the train times and routes. + +Unfortunately for me, although the data is available in the same format, and is open data, it is not always easy to find and there was no single source for the data I needed. This is a place that many people are trying to improve, but it is clearly still a work in progress. + +Instead I found `https://trainmap.ntag.fr` and someone had a differnt approach, instead or relying on the train companies, it uses osm data and osmrouting to generate the routes from the rail network. This approach has its limitations, but it was perfect for quickly solving my use case. + +I wrote a simple python script that looked up the the id of each train station from a [csv by trainline.eu](https://raw.githubusercontent.com/trainline-eu/stations/master/stations.csv) and would try to request the geojson from the api at `https://trainmap.ntag.fr/api/route/?simplify=1&dep={dep_id}&arr={arr_id}` where `dep` and `arr` are the station ids. + +```python + +``` + + + +The returned geojson feature was slightly off, as it always returned a "type: Polygon" instead of "type: LineString" but the data was a legit LineString. So I data wrangled the type and added the structured data to the properties of the geojson feature. + +I did a quick santity check using [geojson.io](geojson.io) and all looked good. + +TODO: add screen shot, or even link to geojson.io + +![alt text](image.png) + +{{Screenshot of geojson.io}} + +### Bike ride data + +To complete this data representation of my journey, I would need to add the sections that I rode my bike. For the sections that I rode my bike, I used Komoot to record the data and downloaded the data as a gpx file. I wanted to convert the `.gpx` file into a `geojson` LineString feature and add it to the feature collection. + +There is a great tool out there called [`gdal`](https://gdal.org) which is described itself as: + +> GDAL is a translator library for raster and vector geospatial data formats + +Sounds perfect, a sword of the geospatial data wrangler! I inspected the gpx files using `ogrinfo` and wrote a little one liner using the [`ogr2ogr`](https://gdal.org/en/latest/faq.html#what-is-this-ogr-stuff) and let the tool do its magic. + +GPX files are, in short, XML files with different tags for different types of geospatial data. + +`` tags are used to represent waypoints +`` tags are used to represent tracks +`` tags are used to represent routes + +I used `ogrinfo` to inspect the data to confirm it only contained a track and no other elements. + +```bash +➜ trains ogrinfo -al -so data/menton_sanremo.gpx +INFO: Open of `data/menton_sanremo.gpx' + using driver `GPX' successful. +Metadata: + AUTHOR_LINK_HREF=https://www.komoot.de + AUTHOR_LINK_TEXT=komoot + AUTHOR_LINK_TYPE=text/html + NAME=Road Ride + +Layer name: waypoints +Geometry: Point +Feature Count: 0 +Layer SRS WKT: +GEOGCRS["WGS 84", + DATUM["World Geodetic System 1984", + ELLIPSOID["WGS 84",6378137,298.257223563, + LENGTHUNIT["metre",1]]], + PRIMEM["Greenwich",0, + ANGLEUNIT["degree",0.0174532925199433]], + CS[ellipsoidal,2], + AXIS["geodetic latitude (Lat)",north, + ORDER[1], + ANGLEUNIT["degree",0.0174532925199433]], + AXIS["geodetic longitude (Lon)",east, + ORDER[2], + ANGLEUNIT["degree",0.0174532925199433]], + ID["EPSG",4326]] +Data axis to CRS axis mapping: 2,1 +ele: Real (0.0) +time: DateTime +magvar: Real (0.0) +geoidheight: Real (0.0) +name: String (0.0) +cmt: String (0.0) +desc: String (0.0) +src: String (0.0) +link1_href: String (0.0) +link1_text: String (0.0) +link1_type: String (0.0) +link2_href: String (0.0) +link2_text: String (0.0) +link2_type: String (0.0) +sym: String (0.0) +type: String (0.0) +fix: String (0.0) +sat: Integer (0.0) +hdop: Real (0.0) +vdop: Real (0.0) +pdop: Real (0.0) +ageofdgpsdata: Real (0.0) +dgpsid: Integer (0.0) + +Layer name: routes +Geometry: Line String +Feature Count: 0 +Layer SRS WKT: +GEOGCRS["WGS 84", + DATUM["World Geodetic System 1984", + ELLIPSOID["WGS 84",6378137,298.257223563, + LENGTHUNIT["metre",1]]], + PRIMEM["Greenwich",0, + ANGLEUNIT["degree",0.0174532925199433]], + CS[ellipsoidal,2], + AXIS["geodetic latitude (Lat)",north, + ORDER[1], + ANGLEUNIT["degree",0.0174532925199433]], + AXIS["geodetic longitude (Lon)",east, + ORDER[2], + ANGLEUNIT["degree",0.0174532925199433]], + ID["EPSG",4326]] +Data axis to CRS axis mapping: 2,1 +name: String (0.0) +cmt: String (0.0) +desc: String (0.0) +src: String (0.0) +link1_href: String (0.0) +link1_text: String (0.0) +link1_type: String (0.0) +link2_href: String (0.0) +link2_text: String (0.0) +link2_type: String (0.0) +number: Integer (0.0) +type: String (0.0) + +Layer name: tracks +Geometry: Multi Line String +Feature Count: 1 +Extent: (7.507404, 43.774250) - (7.784715, 43.822411) +Layer SRS WKT: +GEOGCRS["WGS 84", + DATUM["World Geodetic System 1984", + ELLIPSOID["WGS 84",6378137,298.257223563, + LENGTHUNIT["metre",1]]], + PRIMEM["Greenwich",0, + ANGLEUNIT["degree",0.0174532925199433]], + CS[ellipsoidal,2], + AXIS["geodetic latitude (Lat)",north, + ORDER[1], + ANGLEUNIT["degree",0.0174532925199433]], + AXIS["geodetic longitude (Lon)",east, + ORDER[2], + ANGLEUNIT["degree",0.0174532925199433]], + ID["EPSG",4326]] +Data axis to CRS axis mapping: 2,1 +name: String (0.0) +cmt: String (0.0) +desc: String (0.0) +src: String (0.0) +link1_href: String (0.0) +link1_text: String (0.0) +link1_type: String (0.0) +link2_href: String (0.0) +link2_text: String (0.0) +link2_type: String (0.0) +number: Integer (0.0) +type: String (0.0) + +Layer name: route_points +Geometry: Point +Feature Count: 0 +Layer SRS WKT: +GEOGCRS["WGS 84", + DATUM["World Geodetic System 1984", + ELLIPSOID["WGS 84",6378137,298.257223563, + LENGTHUNIT["metre",1]]], + PRIMEM["Greenwich",0, + ANGLEUNIT["degree",0.0174532925199433]], + CS[ellipsoidal,2], + AXIS["geodetic latitude (Lat)",north, + ORDER[1], + ANGLEUNIT["degree",0.0174532925199433]], + AXIS["geodetic longitude (Lon)",east, + ORDER[2], + ANGLEUNIT["degree",0.0174532925199433]], + ID["EPSG",4326]] +Data axis to CRS axis mapping: 2,1 +route_fid: Integer (0.0) +route_point_id: Integer (0.0) +ele: Real (0.0) +time: DateTime +magvar: Real (0.0) +geoidheight: Real (0.0) +name: String (0.0) +cmt: String (0.0) +desc: String (0.0) +src: String (0.0) +link1_href: String (0.0) +link1_text: String (0.0) +link1_type: String (0.0) +link2_href: String (0.0) +link2_text: String (0.0) +link2_type: String (0.0) +sym: String (0.0) +type: String (0.0) +fix: String (0.0) +sat: Integer (0.0) +hdop: Real (0.0) +vdop: Real (0.0) +pdop: Real (0.0) +ageofdgpsdata: Real (0.0) +dgpsid: Integer (0.0) + +Layer name: track_points +Geometry: Point +Feature Count: 2280 +Extent: (7.507404, 43.774250) - (7.784715, 43.822411) +Layer SRS WKT: +GEOGCRS["WGS 84", + DATUM["World Geodetic System 1984", + ELLIPSOID["WGS 84",6378137,298.257223563, + LENGTHUNIT["metre",1]]], + PRIMEM["Greenwich",0, + ANGLEUNIT["degree",0.0174532925199433]], + CS[ellipsoidal,2], + AXIS["geodetic latitude (Lat)",north, + ORDER[1], + ANGLEUNIT["degree",0.0174532925199433]], + AXIS["geodetic longitude (Lon)",east, + ORDER[2], + ANGLEUNIT["degree",0.0174532925199433]], + ID["EPSG",4326]] +Data axis to CRS axis mapping: 2,1 +track_fid: Integer (0.0) +track_seg_id: Integer (0.0) +track_seg_point_id: Integer (0.0) +ele: Real (0.0) +time: DateTime +magvar: Real (0.0) +geoidheight: Real (0.0) +name: String (0.0) +cmt: String (0.0) +desc: String (0.0) +src: String (0.0) +link1_href: String (0.0) +link1_text: String (0.0) +link1_type: String (0.0) +link2_href: String (0.0) +link2_text: String (0.0) +link2_type: String (0.0) +sym: String (0.0) +type: String (0.0) +fix: String (0.0) +sat: Integer (0.0) +hdop: Real (0.0) +vdop: Real (0.0) +pdop: Real (0.0) +ageofdgpsdata: Real (0.0) +dgpsid: Integer (0.0) +``` + +Seems legit, the `tracks` layer contains the data I want. I can now convert this to a `geojson` file using the `ogr2ogr` command line tool. + +```bash +for file in data/*.gpx; do ogr2ogr -f "GeoJSON" -nlt LINESTRING -sql "SELECT *, 'yes' AS bike FROM tracks" "${file%.gpx}.geojson" "$file"; done +``` + +This command loops through all the `.gpx` files in the `data` directory and converts them to `geojson` files with the same name. + +The sql query `-sql "SELECT *, 'yes' AS bike FROM tracks"` selects all the data from the `tracks` layer which referes to the `` elements in the `.gpx` file. It will take any of the attributes from the `` element and add them to the `properties` of the `geojson` feature, in our case thats the name. I +also add a new field called `bike` with the value `yes` to indicate that this is a bike ride. + +The `-nlt LINESTRING` flag tells `ogr2ogr` to convert the data to a `LineString` feature, if I didn't include this flag the data would be converted to a `MultiLineString` feature which is not what I want. + +There is many ways to achieve the above data processing, and a vast amount of open tools out there to help achieve pretty much anything you want. Long live Open Data and open source software! + +Finally I used [maplibre-gl](https://maplibre.org/) to visualize the data. My idea was to make a little static webpage with I wanted the features: + +- journey legs displayed on a color coded line depending on the train company or if it was cycled +- Hover actionss +- Pop up displaying information about the leg + - Train number + - Train type + - Train operator + - bike rating + - wifi + - price + +I used webpack to bundle the little website up and then deployed it using the classic github pages. + +To embed the map into this blog post, I added a iframe and set the src to point to the github pages url. + +```html + +``` + +{{}} + +{{}} +``` + +## Totals and stats + +After adding together how much it cost it took me + +€ TODO: add this shit up. +Hours moving in trains +km on the bike +4 espressos +TODO: add some stats, total rivers crossed, tunnels went through, pizza eaten, +Crashes + diff --git a/static/media/kreuzungen/Openstreetmap_logo.svg b/static/media/kreuzungen/Openstreetmap_logo.svg new file mode 100644 index 0000000..db5a4eb --- /dev/null +++ b/static/media/kreuzungen/Openstreetmap_logo.svg @@ -0,0 +1,277 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + 010110010011010110010011 + 010110010011010110010011 + + \ No newline at end of file diff --git a/static/media/kreuzungen/dickemarie.jpg b/static/media/kreuzungen/dickemarie.jpg new file mode 100644 index 0000000..1837ace Binary files /dev/null and b/static/media/kreuzungen/dickemarie.jpg differ diff --git a/static/media/kreuzungen/domain_loading.mp4 b/static/media/kreuzungen/domain_loading.mp4 new file mode 100644 index 0000000..e69de29 diff --git a/static/media/kreuzungen/domain_loading.webm b/static/media/kreuzungen/domain_loading.webm new file mode 100644 index 0000000..a94bf45 Binary files /dev/null and b/static/media/kreuzungen/domain_loading.webm differ diff --git a/static/media/kreuzungen/github_pages_deploy.png b/static/media/kreuzungen/github_pages_deploy.png new file mode 100644 index 0000000..74f8cba Binary files /dev/null and b/static/media/kreuzungen/github_pages_deploy.png differ diff --git a/static/media/kreuzungen/google.png b/static/media/kreuzungen/google.png new file mode 100644 index 0000000..6fefc5b Binary files /dev/null and b/static/media/kreuzungen/google.png differ diff --git a/static/media/kreuzungen/komoot_screenshot.png b/static/media/kreuzungen/komoot_screenshot.png new file mode 100644 index 0000000..7a45abb Binary files /dev/null and b/static/media/kreuzungen/komoot_screenshot.png differ diff --git a/static/media/kreuzungen/loadingkreuzungen.gif b/static/media/kreuzungen/loadingkreuzungen.gif new file mode 100644 index 0000000..164267f Binary files /dev/null and b/static/media/kreuzungen/loadingkreuzungen.gif differ diff --git a/static/media/kreuzungen/oil_crossings.jpeg b/static/media/kreuzungen/oil_crossings.jpeg new file mode 100644 index 0000000..eaeee53 Binary files /dev/null and b/static/media/kreuzungen/oil_crossings.jpeg differ diff --git a/static/media/kreuzungen/oslogo.svg.png b/static/media/kreuzungen/oslogo.svg.png new file mode 100644 index 0000000..0491819 Binary files /dev/null and b/static/media/kreuzungen/oslogo.svg.png differ diff --git a/static/media/kreuzungen/overpassturbo.png b/static/media/kreuzungen/overpassturbo.png new file mode 100644 index 0000000..41f98dc Binary files /dev/null and b/static/media/kreuzungen/overpassturbo.png differ diff --git a/static/media/kreuzungen/screenshot_frame.png b/static/media/kreuzungen/screenshot_frame.png new file mode 100644 index 0000000..718bdaa Binary files /dev/null and b/static/media/kreuzungen/screenshot_frame.png differ diff --git a/static/media/kreuzungen/whatsapp.gif b/static/media/kreuzungen/whatsapp.gif new file mode 100644 index 0000000..7e601c1 Binary files /dev/null and b/static/media/kreuzungen/whatsapp.gif differ diff --git a/static/media/kreuzungen/zwift_screenshot.png b/static/media/kreuzungen/zwift_screenshot.png new file mode 100644 index 0000000..97d6a21 Binary files /dev/null and b/static/media/kreuzungen/zwift_screenshot.png differ diff --git a/static/media/kreuzungen/zwifthacks.svg b/static/media/kreuzungen/zwifthacks.svg new file mode 100644 index 0000000..7d9cfe6 --- /dev/null +++ b/static/media/kreuzungen/zwifthacks.svg @@ -0,0 +1,26799 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +NY Sprint +NY Sprint + + +Central Park Loop +Central Park Loop + + + + + + + +17 + + + + + +NY Sprint + + +Central Park Loop + + + + +NY SPRINT + + + + + + + + + + + + + + + + + + +17 + + + + +NY Sprint +NY Sprint + + + +Central Park Loop +Central Park Loop + + + + + + + + + + + + + + + + + +Sorry, your browser does not support inline SVG. + \ No newline at end of file diff --git a/static/media/trains/geojson_io.png b/static/media/trains/geojson_io.png new file mode 100644 index 0000000..5d8430d Binary files /dev/null and b/static/media/trains/geojson_io.png differ diff --git a/static/media/trains/google_maps.png b/static/media/trains/google_maps.png new file mode 100644 index 0000000..fc0eb39 Binary files /dev/null and b/static/media/trains/google_maps.png differ diff --git a/static/media/trains/leportail.png b/static/media/trains/leportail.png new file mode 100644 index 0000000..1716db8 Binary files /dev/null and b/static/media/trains/leportail.png differ