-
-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'main' into blog/qchat-news
- Loading branch information
Showing
7 changed files
with
356 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,250 @@ | ||
--- | ||
authors: | ||
- ffougeres | ||
categories: | ||
- Article | ||
comments: true | ||
date: 2024-12-02 | ||
description: Create and visualize GIS data on the progress of the Vendée Globe 2024 race from official spreadsheets. | ||
icon: material/sail-boat | ||
license: beerware | ||
links: | ||
- French version: https://geotribu.fr/articles/2024/2024-11-20_vendee_globe_donnees_sig/ | ||
- Official race website: https://www.vendeeglobe.org/en | ||
tags: | ||
- GeoPandas | ||
- Pandas | ||
- Python | ||
- QGIS | ||
- Vendée Globe | ||
- sailing | ||
title: Track the Vendée Globe race with Python and QGIS | ||
subtitle: Sailing GIS | ||
--- | ||
|
||
# Follow the Vendée Globe 2024 from a GIS | ||
|
||
## What is the Vendée Globe? | ||
|
||
![logo Vendée Globe](https://cdn.geotribu.fr/img/logos-icones/divers/vendee_globe.png){: .img-thumbnail-left } | ||
|
||
Before we start talking about GIS and technical aspects, let's talk about the Vendée Globe. | ||
|
||
It is a solo sailing race, non-stop and without assistance, around the world. It has been held every 4 years since 1989. The start is in Les Sables d'Olonne. The course consists of going down the Atlantic, then passing successively under Africa and the Cape of Good Hope, under Australia and Cape Leeuwin and finally under South America and Cape Horn, to return to Vendée as quickly as possible. The record was set by Armel Le Cléac'h during the 2016-2017 edition with a journey of 74 days 3 hours and 35 minutes. | ||
|
||
![map](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/carte_vendee_globe.svg){: .img-center loading=lazy } | ||
|
||
<!-- more --> | ||
|
||
---- | ||
|
||
## Follow the progress | ||
|
||
![Smarty Pins logo](https://cdn.geotribu.fr/img/logos-icones/entreprises_association/google/SmartyPins.png){: .img-thumbnail-left } | ||
|
||
Who says race around the world, necessarily says map to follow the progress of the participants. The official website of the event offers an [interactive map](https://www.vendeeglobe.org/cartographie) to visualize this progress. | ||
|
||
![Vendée Globe - Official interactive map](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/carte_interactive.png){: .img-center loading=lazy } | ||
|
||
So I looked to see if there was an API or web service providing the positioning data to visualize them in a GIS, like QGIS for example. After some research, I found nothing like that. | ||
|
||
I found a [discussion](https://www.reddit.com/r/Vendee_Globe/s/Gbli34xyQO) on Reddit about this, but without a conclusive answer. | ||
|
||
On the other hand, I ended up discovering that the official website publishes every 4 hours an Excel file containing the navigation data and the coordinates of the boats. | ||
|
||
![Vendée Globe - Tableur des données de navigation](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/tableur.png){: .img-center loading=lazy } | ||
|
||
This file communicates the positions every day at 2am, 6am, 10am, 2pm, 6pm and 10pm, with a delay of 1 hour. For example, the 10am file is provided at 11am (this is an element that will be taken into account in the industrialization of the process). To download this file, go to the [ranking](https://www.vendeeglobe.org/classement) section. | ||
|
||
This table contains the rank, the name of the boat and the skipper, but also the speed and course over the last 30 minutes, the last 24 hours and since the last score. | ||
|
||
From this spreadsheet, the goal will therefore be to build geographic data for the race, whether to trace the trajectory, but also to aggregate all the scores. | ||
|
||
[Comment on this article :fontawesome-solid-comments:](#__comments "Go to comments"){: .md-button } | ||
{: align=middle } | ||
|
||
---- | ||
|
||
## Steps to follow | ||
|
||
You must start by retrieving the information relating to the positions of the boats. This means downloading the Excel files, because the site does not allow you to retrieve them in bulk. So I studied the structure of the URL to understand how they were generated and thus be able to reconstruct these download links. | ||
|
||
```shell title="Format of the download URL of the scoreboard" | ||
https://www.vendeeglobe.org/sites/default/files/ranking/vendeeglobe_leaderboard_YYYYMMDD_HHMMSS.xlsx | ||
``` | ||
|
||
It is therefore necessary to complete the date (YYYYMMDD format) and the time of the score (HHMMSS) to construct the download URL. | ||
|
||
Then, it is necessary to deal with the way in which the location data is presented. Indeed, the positions of the boats are often provided in a geographic coordinate format in degrees, minutes and seconds (DMS). Although this format is useful, it is not directly compatible with geomatics tools. It is therefore essential to convert them into decimal degrees, a more standard and precise format, which allows easy working with maps and geographic information systems (GIS). | ||
|
||
Finally, it is important to export this GIS data in a compatible format, such as GeoPackage or GeoJSON. Once converted, this data can be used in any GIS, whether it is a desktop GIS like QGIS or a web GIS map with tools like MapLibre or Leaflet. | ||
|
||
---- | ||
|
||
## Industrialize the method | ||
|
||
![factory logo](https://cdn.geotribu.fr/img/logos-icones/divers/factory.png){: .img-thumbnail-left } | ||
|
||
To automate the process described above, I created a [GitHub project](https://github.com/florentfgrs/Vendee-Globe-2024) that automates these tasks with Python scripts. It works in command lines, and there are currently two of them (see below). | ||
|
||
For downloading, I use the third-party library `requests`. | ||
|
||
For reading the spreadsheet, cleaning the data and creating geometry, I use `pandas`, `geopandas` and `shapely`. There is some data cleaning to do, because the cells contain line breaks. | ||
|
||
To go into more technical detail, once the file is downloaded, the successive steps are: | ||
|
||
1. **Opening the file in a dataframe, keeping only the columns and rows that interest us.** | ||
This involves loading the Excel file and extracting it into a dataframe, keeping only the relevant data for further processing, while ignoring the superfluous information. | ||
|
||
2. **Creating headers.** | ||
The headers of the Excel file are often made up of merged cells, which makes them difficult to retrieve. In addition, the column names are sometimes too verbose, so they need to be simplified to make them more usable. | ||
|
||
3. **Data cleaning.** | ||
This step involves removing line breaks, special characters, or any other anomalies that could disrupt data processing. | ||
|
||
```pandas title="DataFrame before cleaning" | ||
rang code nom heure ... 24h_vmg 24h_distance dtf dtl | ||
0 1 GBR\r\nFRA 100 Sam Goodchild\r\nVULNERABLE 10:30 FR\r\n ... 10.5 kts 255.1 nm 22300.7 nm 0.0 nm | ||
1 2 FRA\r\nFRA 112 Sébastien Simon\r\nGroupe Dubreuil 10:30 FR\r\n ... 7.4 kts 223.1 nm 22324.7 nm 24.0 nm | ||
2 3 FRA\r\nFRA 59 Thomas Ruyant\r\nVULNERABLE 10:30 FR\r\n ... 10.7 kts 288.1 nm 22352.7 nm 52.0 nm | ||
3 4 FRA\r\nFRA85 Nicolas Lunven\r\nHOLCIM - PRB 10:30 FR\r\n ... 12.7 kts 306.4 nm 22378.5 nm 77.8 nm | ||
4 5 FRA\r\nFRA 29 Jean Le Cam\r\nTout commence en Finistère - Ar... 10:30 FR\r\n ... 5.0 kts 158.5 nm 22379.0 nm 78.3 nm | ||
5 6 FRA\r\nFRA 15 Clarisse Crémer\r\nL'Occitane en Provence 10:30 FR\r\n ... 7.3 kts 211.9 nm 22410.7 nm 110.1 nm | ||
``` | ||
|
||
```pandas title="DataFrame after cleaning" | ||
rang code nom heure ... 24h_vmg 24h_distance dtf dtl | ||
0 1 GBR - FRA 100 Sam Goodchild - VULNERABLE 10:30 FR - ... 10.5 kts 255.1 nm 22300.7 nm 0.0 nm | ||
1 2 FRA - FRA 112 Sébastien Simon - Groupe Dubreuil 10:30 FR - ... 7.4 kts 223.1 nm 22324.7 nm 24.0 nm | ||
2 3 FRA - FRA 59 Thomas Ruyant - VULNERABLE 10:30 FR - ... 10.7 kts 288.1 nm 22352.7 nm 52.0 nm | ||
3 4 FRA - FRA85 Nicolas Lunven - HOLCIM - PRB 10:30 FR - ... 12.7 kts 306.4 nm 22378.5 nm 77.8 nm | ||
4 5 FRA - FRA 29 Jean Le Cam - Tout commence en Finistère - Arm... 10:30 FR - ... 5.0 kts 158.5 nm 22379.0 nm 78.3 nm | ||
5 6 FRA - FRA 15 Clarisse Crémer - L'Occitane en Provence 10:30 FR - ... 7.3 kts 211.9 nm 22410.7 nm 110.1 nm | ||
``` | ||
|
||
4. **Creating the timestamp.** | ||
A timestamp must be generated for each pointing in order to track the evolution of the position of the boats over time. It will also be useful for building the trajectory. | ||
|
||
```pandas title="Creating timestamp column from excel file time and date column" | ||
heure timestamp | ||
0 10:30 FR - 2024-11-18 10:30:00 | ||
1 10:30 FR - 2024-11-18 10:30:00 | ||
2 10:30 FR - 2024-11-18 10:30:00 | ||
3 10:30 FR - 2024-11-18 10:30:00 | ||
4 10:30 FR - 2024-11-18 10:30:00 | ||
5 10:30 FR - 2024-11-18 10:30:00 | ||
``` | ||
|
||
5. **Converting latitude and longitude columns from DMS degrees to decimal degrees.** | ||
First, you need to parse the coordinates to get the degrees, minutes, seconds and orientation. Then do the conversion. | ||
|
||
6. **Geometry creation.** | ||
From the converted coordinates, you need to generate geometries. This consists of generating points for the positions of the boats (when plotting) or lines to draw the trajectories. | ||
|
||
```pandas title="Converting DMS latitude/longitude to decimal then creating the geometry column" | ||
latitude longitude latitude_decimal longitude_decimal geometry | ||
0 17°56.15'N 31°09.06'W 17.937500 -31.151667 POINT (-31.15167 17.9375) | ||
1 18°32.68'N 30°10.63'W 18.552222 -30.184167 POINT (-30.18417 18.55222) | ||
2 18°19.45'N 33°17.34'W 18.329167 -33.292778 POINT (-33.29278 18.32917) | ||
3 18°59.38'N 32°23.11'W 18.993889 -32.386389 POINT (-32.38639 18.99389) | ||
4 19°17.37'N 19°24.52'W 19.293611 -19.414444 POINT (-19.41444 19.29361) | ||
5 19°58.12'N 30°22.88'W 19.970000 -30.391111 POINT (-30.39111 19.97) | ||
``` | ||
|
||
7. **Export to vector GIS format.** | ||
Export to [Geopackage](https://www.geopackage.org/) format. | ||
|
||
For now, this project offers two features: | ||
|
||
### Get last point | ||
|
||
It This is a layer of points indicating the last communicated position of the competitors. The resulting format is a geopackage. | ||
|
||
```shell title="Get a GPKG with the latest pointing date" | ||
python latest_pointing.py --output-dir ./data_vg | ||
``` | ||
|
||
The result obtained is a layer of points of the latest pointing date. For example, if I run this command line at 2:45 p.m., I will have the 10 a.m. time stamp (and not the 2 p.m. one because of the 1-hour publication delay). | ||
|
||
Once displayed in QGIS and with a little work on the style, here is the result: | ||
|
||
![Screenshot QGIS - Layer of positions of the last pointing](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/dernier_pointage.png){: .img-center loading=lazy } | ||
|
||
### Get all the scores and the track from the start | ||
|
||
This is a layer of points indicating all the scores of each boat from the start, as well as a layer of lines connecting these points to form the trajectory of the boats. The resulting format is also a GeoPackage. | ||
|
||
```shell title="Obtain a GPKG with the entire track and pointages" | ||
python trajectoires_pointages.py --output-dir ./data_vg | ||
``` | ||
|
||
We obtain a geopackage which contains two layers: | ||
|
||
- A layer of the history of all the pointings since the start. | ||
- A line layer that reproduces the trajectory of each boat. | ||
|
||
![Screenshot QGIS - Layers of the trajectories since the start and all the points](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/trajectoire.png){: .img-center loading=lazy } | ||
|
||
### Attribute data | ||
|
||
In both features, we find in the attribute table of the layers all the information of the spreadsheet. I only added a `timestamp` column, it is used to link the points together and create the trajectory layer. | ||
|
||
![Screenshot QGIS - Data attribute table](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/table_attrib.png){: .img-center loading=lazy } | ||
|
||
!!! info "Meaning of prefixes in column names" | ||
- `30m` = Since 30 minutes | ||
- `last_rank` = Since the previous point | ||
- `24h` = Since 24h | ||
|
||
Maybe we should remove the units in the data to have numerical values? In this case, we should perhaps add the units in the column names. This is one of the avenues for improvement. I would also like to separate the skipper's name and the boat's name into two separate columns. Contributions to improve this code are welcome. | ||
|
||
---- | ||
|
||
## Animate the progression with QGIS's Temporal Control | ||
|
||
![logo QGIS](https://cdn.geotribu.fr/img/logos-icones/logiciels_librairies/qgis.png "logo QGIS"){: .img-thumbnail-left } | ||
|
||
To visualize the data, QGIS is ideal and since the data has a temporal dimension, it is an opportunity to play with the temporal controller. | ||
For this tutorial, you must use the `pointages` layer produced by `trajectoires_pointages.py`. | ||
|
||
### Configure the layer | ||
|
||
After accessing the layer properties (right-click > Properties), go to the **Temporal** tab. Configure the settings as follows: | ||
|
||
![QGIS - Temporal tab configuration](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/temporel.png){: .img-center loading=lazy } | ||
|
||
### Show the temporal toolbar | ||
|
||
- Right-click at the top of the toolbars. | ||
- Check (if not already done) **Temporal controller panel** in the **Panels** section. | ||
|
||
### Configure the toolbar | ||
|
||
- Adjust the start date at the start of the event. | ||
- Indicate a step of 4 hours (this is the delta between two points). | ||
|
||
![QGIS - Temporal Controller Configuration](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/controleur.png){: .img-center loading=lazy } | ||
|
||
### Layer animation | ||
|
||
After clicking Play, here is the result you should get: | ||
|
||
![QGIS - Temporal Controller Animation](https://cdn.geotribu.fr/img/articles-blog-rdp/articles/2024/vendee_globe_donnees_sig/qgis-temporal.gif){: .img-center loading=lazy } | ||
|
||
<!-- markdownlint-disable MD046 --> | ||
!!! tip "QGIS expression to filter tracking on a competitor" | ||
|
||
```sql | ||
"skipper" = 'Maxime Sorel' | ||
``` | ||
<!-- markdownlint-enable MD046 --> | ||
|
||
## To go further | ||
|
||
This first step is only a POC (Proof of Concept) the code can still be optimized and I will continue to do so throughout the race (hoping that the formalism and publication times of the spreadsheet do not change). Subsequently, several ideas could be explored. I will surely explore one of them. | ||
|
||
- **Create a QGIS plugin**: A QGIS plugin could allow to load the ranking, the last position of the ships, and their trajectory. We could imagine that the post-processing of the Excel file to GIS data is done by continuous integration (CI) and exported in GeoJSON, and that the plugin loads these GeoJSON hosted in the GitHub project. | ||
|
||
- **Provide data via an API**: We could imagine a project that automatically retrieves this data, converts and structures it, then exposes an API that provides a position or trajectory based on a competitor's number, for example. | ||
|
||
- **Create a cartographic web application** to visualize the progress of the boats with more possibilities than what the official cartographic interface offers. I had imagined using [mviewer](https://mviewer.github.io/fr/) for this. |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,95 @@ | ||
#! python3 # noqa: E265 | ||
|
||
"""Insert social share buttons in content. | ||
Related resources: | ||
- https://squidfunk.github.io/mkdocs-material/tutorials/blogs/engage/ | ||
- https://stackoverflow.com/questions/10713542/how-to-make-a-custom-linkedin-share-button | ||
- https://stackoverflow.com/questions/24823114/post-to-reddit-via-url | ||
- https://docs.bsky.app/docs/advanced-guides/intent-links | ||
""" | ||
|
||
# ############################################################################ | ||
# ########## Libraries ############# | ||
# ################################## | ||
|
||
# standard library | ||
import logging | ||
import re | ||
import urllib.parse | ||
from textwrap import dedent | ||
|
||
# Mkdocs | ||
import mkdocs.plugins | ||
from mkdocs.config.defaults import MkDocsConfig | ||
from mkdocs.structure.pages import Page | ||
|
||
# ########################################################################### | ||
# ########## Global ################ | ||
# ################################## | ||
|
||
|
||
logger = logging.getLogger("mkdocs") | ||
|
||
regex_pattern_include = re.compile(r"posts/[1-9].*") | ||
|
||
share_url_base_bluesky = "https://bsky.app/intent/compose" | ||
share_url_base_mastodon = "https://mastodonshare.com/share" | ||
share_url_base_linkedin = "https://www.linkedin.com/shareArticle" | ||
share_url_base_reddit = "http://www.reddit.com/submit" | ||
|
||
# ########################################################################### | ||
# ########## Functions ############# | ||
# ################################## | ||
|
||
|
||
@mkdocs.plugins.event_priority(-100) | ||
def on_page_markdown(markdown: str, page: Page, config: MkDocsConfig, **kwargs): | ||
|
||
if not regex_pattern_include.match(page.file.src_uri): | ||
return markdown | ||
|
||
# prepare data | ||
page_authors = urllib.parse.quote(", ".join([a.get("name") for a in page.authors])) | ||
page_description = urllib.parse.quote(page.meta.get("description") + "\n") | ||
page_tags = "%20".join( | ||
[ | ||
urllib.parse.quote(f"#{tag.replace(' ', '')}") | ||
for tag in page.meta.get("tags") | ||
] | ||
) | ||
|
||
page_title = urllib.parse.quote(page.title + "\n") | ||
page_url = config.site_url + page.url | ||
|
||
# urls | ||
share_url_bluesky = ( | ||
f"{share_url_base_bluesky}?text=" | ||
+ f"{page_title}%0A{page_description}%0Aby%20{page_authors}.%20%0A%0A%0A%23GISTribe%0A%0ACheck%20it%20on%[email protected]%20:%20{page_url}" | ||
) | ||
share_url_linkedin = ( | ||
f"{share_url_base_linkedin}?url={page_url}" | ||
"&mini=true" | ||
# f"&title={page_title}" | ||
# f"&summary={page_description}" | ||
f"&source=Geotribu" | ||
f"&text=Check%20this%20blog%20post%20{page_title}%20by%20{page_authors}%20" | ||
"on%20%40Geotribu%0A" | ||
f"{page_description}%0A%0A%20" | ||
f"%0A%23GISTribe%20{page_tags}" | ||
) | ||
share_url_mastodon = ( | ||
f"{share_url_base_mastodon}?url={page_url}" | ||
f"&text=Check%20this%20blog%20post%20{page_title}%20by%20{page_authors}%20" | ||
"on%20%40geotribu%40mapstodon.space%0A" | ||
f"{page_description}%0A%0A%20" | ||
# f"%0A%23GISTribe%20{page_tags}" | ||
) | ||
share_url_reddit = f"{share_url_base_reddit}?title={page_title}%20by%20{page_authors}&url={page_url}" | ||
|
||
return markdown + dedent( | ||
f"""\n\nShare it: [:fontawesome-brands-bluesky:]({share_url_bluesky}) [:fontawesome-brands-linkedin:]({share_url_linkedin}) [:fontawesome-brands-mastodon:]({share_url_mastodon}) [:fontawesome-brands-reddit:]({share_url_reddit}) | ||
{{: align=middle .md-source }} | ||
""" | ||
) |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters