--- title: "Codebook and cookbook for v2 (2022 onwards) Spanish mobility data" vignette: > %\VignetteIndexEntry{Codebook and cookbook for v2 (2022 onwards) Spanish mobility data} %\VignetteEngine{quarto::html} %\VignetteEncoding{UTF-8} bibliography: references.bib format: html: toc: true toc-depth: 2 code-overflow: wrap execute: eval: false --- You can view this vignette any time by running: ```{r} spanishoddata::spod_codebook(ver = 2) ``` The mobility data v2 (2022 onwards) was originally released by the Ministerio de Transportes, Movilidad y Agenda Urbana (MITMA) , now [Ministerio de Transportes y Movilidad Sostenible (MITMS)](https://www.transportes.gob.es/) [@mitma_mobility_2024_v8]. The dataset is produced by [Nommon](https://www.nommon.es/){target="_blank"} using the raw data from [Orange España](https://www.orange.es/){target="_blank"}. Even though the raw data is only from one mobile phone operator, the resulting flows and other counts of number of individuals in the data set are already resampled to be representative of the total population of Spain (see details in the official methodology). The tables in the data set provide hourly flows between zones across Spain for every day of the observation period (2022-01-01 onwards), the number of individuals making trips for each zone, the number of individuals spending the nights in one location while regularly residing in a different location, and many more advanced datasets. This document will introduce you to the available data and provide brief code snippets on how to access it using the `{spanishoddata}` R package. Compared to the [v1 data (2020-2021)](v1-2020-2021-mitma-data-codebook.qmd) this dataset has many additional variables, such as age, sex, and income, has better spatial resolution (the zones are more spatially granular) and covers a continuous period[^1] (2022-01-01 onward), rather than only a limited period (in v1 - 2020-02-14 to 2021-05-09). [^1]: For reference: this object also has classes: `tbl_dbi` ,`tbl_sql`, `tbl_lazy` ,and `tbl` . {{< include ../inst/vignette-include/missing-dates-outages.qmd >}} In v2 data, the datasets comparable to the [v1 (2020-2021) data](v1-2020-2021-mitma-data-codebook.qmd) (hourly daily origin destination matrices and number of trips at each location) are called ["basic studies" ("estudios basicos")](https://www.transportes.gob.es/ministerio/proyectos-singulares/estudios-de-movilidad-con-big-data/estudio-basico-diario){target="_blank"}, however there are even more advanced datasets that are already available or will be made available soon: ["complete studies" ("estudios completos")](https://www.transportes.gob.es/ministerio/proyectos-singulares/estudios-de-movilidad-con-big-data/estudios-completos){target="_blank"} and ["road routes" ("rutas de carretera")](https://www.transportes.gob.es/ministerio/proyectos-singulares/estudios-de-movilidad-con-big-data/rutas-por-carretera){target="_blank"}. At the moment, the `{spanishoddata}` package only provides the interface to the "basic studies" datasets, but support for "complete studies" and "road routes" will be added in the future. Key sources for this codebook/cookbook include: - [original data collection methodology and codebook in Spanish](https://www.transportes.gob.es/recursos_mfom/paginabasica/recursos/a3_informe_metodologico_estudio_movilidad_mitms_v8.pdf){target="_blank"} + [automatically translated English version of methodology and codebook](https://rOpenSpain.github.io/spanishoddata/codebooks/a3_informe_metodologico_estudio_movilidad_mitms_v8_en.pdf){target="_blank"} - [original quality assessment](https://www.transportes.gob.es/recursos_mfom/paginabasica/recursos/20241024_validaciones_estudios_basicos_bigdata_v1.0.pdf){target="_blank"} + [automatically translated English version of quality assessment](https://rOpenSpain.github.io/spanishoddata/codebooks/20241024_validaciones_estudios_basicos_bigdata_v1.0_en.pdf){target="_blank"} - [original data download page](https://www.transportes.gob.es/ministerio/proyectos-singulares/estudios-de-movilidad-con-big-data/opendata-movilidad){target="_blank"} - [original data license](https://www.transportes.gob.es/el-ministerio/buen-gobierno/licencia_datos){target="_blank"} - [homepage of the v2 study and open mobility data project of the Ministry of Transport and Sustainable Mobility of Spain](https://www.transportes.gob.es/ministerio/proyectos-singulares/estudio-de-movilidad-con-big-data){target="_blank"} ::: callout-note Kindly consult the documents above for any specific details on the methodology. The codebook here is only a simplified summary. ::: To access the data we reference in this codebook, please follow these steps: {{< include ../inst/vignette-include/install-package.qmd >}} Load it as follows: ```{r} library(spanishoddata) ``` Using the instructions below, set the data folder for the package to download the files into. You may need up to 400 GB to download all data and another 400 GB if you would like to convert the downloaded data into analysis ready format (a `DuckDB` database file, or a folder of `parquet` files). You can find more info on this conversion in the [Download and convert OD datasets](convert.html) vignette. {{< include ../inst/vignette-include/setup-data-directory.qmd >}} {{< include ../inst/vignette-include/overall-approach.qmd >}} ![The overview of package functions to get the data](../man/figures/package-functions-overview.svg){#fig-overall-flow width="78%"} # 1. Spatial data with zoning boundaries The boundary data is provided at three geographic levels: [`Distrtics`](#districts), [`Municipalities`](#municipalities), and [`Large Urban Areas`](#lua). It's important to note that these do not always align with the official Spanish census districts and municipalities. To comply with data protection regulations, certain aggregations had to be made to districts and municipalities". ## 1.1 `Districts` {#districts} `Districts` correspond to official census districts in cities; however, in those with lower population density, they are grouped together. In rural areas, one district is often equal to a municipality, but municipalities with low population are combined into larger units to preserve privacy of individuals in the dataset. Therefore, there are 3792 'districts' compared to the 10494 official census districts on which they are based. There are also [NUTS3 statistical regions](https://ec.europa.eu/eurostat/web/nuts){target="_blank"} covering France (94 units) and Portugal (23 units). Therefore there is a total of 3909 zones in the `Districts` dataset. ```{r} districts_v2 <- spod_get_zones("dist", ver = 2) ``` The `districts_v2` object is of class `sf` consisting of polygons. Data structure: | Variable Name | **Description** | |------------------|------------------------------------------------------| | `id` | District `id` assigned by the data provider. Matches with `id_origin`, `id_destination`, and `id` in district-level origin-destination and number of trips data. | | `name` | Name of the district. | | `population` | Number of individuals in the district according to [INE](https://ine.es/){target="_blank"}[^2]. | | `census_sections` | Semicolon-separated list of census section identifiers that correspond to each district, classified by the Spanish Statistical Office (INE). | | `census_districts` | Semicolon-separated list of census district identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `municipalities` | Semicolon-separated list of municipality identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `municipalities_mitma` | Semicolon-separated list of municipality identifiers as assigned by the data provider (MITMA). | | `luas_mitma` | Semicolon-separated list of Large Urban Areas (LUAs) as assigned by the data provider, corresponding to each district. | | `district_ids_in_v1` | Semicolon-separated district identifiers from v1 data corresponding to each district in v2. If no match exists, marked as `NA`. | | `geometry` | Spatial geometry of each district stored as a `MULTIPOLYGON` object, projected in the ETRS89 / UTM zone 30N CRS with XY dimensions. | [^2]: This is likely the population as of end of 2021 or start of 2022. Population for a few districts is missing. Instead of population, residence and overnight stays data may be used as a proxy with caution. Also, newer population figures may be obtained and joined with the provided zones using the reference tables that match the zones ids with official municipal and census district ids from INE. ## 1.2 `Municipalities` {#municipalities} `Municipalities` are made up of official municipalities in those of a certain size; however, they have also been aggregated in cases of lower population density. As a result, there are 2618 municipalities compared to the 8,125 official municipalities on which they are based. There are also [NUTS3 statistical regions](https://ec.europa.eu/eurostat/web/nuts){target="_blank"} covering France (94 units) and Portugal (23 units). Therefore there is a total of 2735 zones in the `Districts` dataset. ```{r} municipalities_v2 <- spod_get_zones("muni", ver = 2) ``` The resulting `municipalities_v2` object is type `sf` consisting of polygons. Data structure: | Variable Name | **Description** | |------------------|------------------------------------------------------| | `id` | District `id` assigned by the data provider. Matches with `id_origin`, `id_destination`, and `id` in district-level origin-destination and number of trips data. | | `name` | Name of the district. | | `population` | Number of individuals in the district according to [INE](https://ine.es/){target="_blank"}[^3]. | | `census_sections` | Semicolon-separated list of census section identifiers that correspond to each district, classified by the Spanish Statistical Office (INE). | | `census_districts` | Semicolon-separated list of census district identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `municipalities` | Semicolon-separated list of municipality identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `districts_mitma` | Semicolon-separated list of district identifiers as assigned by the data provider (MITMA). | | `luas_mitma` | Semicolon-separated list of Large Urban Areas (LUAs) as assigned by the data provider, corresponding to each district. | | `municipality_ids_in_v1` | Semicolon-separated district identifiers from v1 data corresponding to each district in v2. If no match exists, marked as `NA`. | | `geometry` | Spatial geometry of each district stored as a `MULTIPOLYGON` object, projected in the ETRS89 / UTM zone 30N CRS with XY dimensions. | [^3]: This is likely the population as of end of 2021 or start of 2022. Population for a few districts is missing. Instead of population, residence and overnight stays data may be used as a proxy with caution. Also, newer population figures may be obtained and joined with the provided zones using the reference tables that match the zones ids with official municipal and census district ids from INE. ## 1.3 `LUAs (Large Urban Areas)` {#luas} `Large Urban Areas (LUAs)` has essentially the same spatial units as [`Municipalities`](#municipalities), but are not aggregated. Therefore, there are 2086 locations in the `LUAs` dataset. There are also [NUTS3 statistical regions](https://ec.europa.eu/eurostat/web/nuts){target="_blank"} covering France (94 units) and Portugal (23 units). Therefore there is a total of 2203 zones in the `LUAs` dataset. ```{r} luas_v2 <- spod_get_zones("lua", ver = 2) ``` The resulting `luas_v2` object is type `sf` consisting of polygons. Data structure: | Variable Name | **Description** | |------------------|------------------------------------------------------| | `id` | District `id` assigned by the data provider. Matches with `id_origin`, `id_destination`, and `id` in district-level origin-destination and number of trips data. | | `name` | Name of the district. | | `population` | Number of individuals in the district according to [INE](https://ine.es/){target="_blank"}[^4]. | | `census_sections` | Semicolon-separated list of census section identifiers that correspond to each district, classified by the Spanish Statistical Office (INE). | | `census_districts` | Semicolon-separated list of census district identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `municipalities` | Semicolon-separated list of municipality identifiers corresponding to each district, as classified by the Spanish Statistical Office (INE). | | `districts_mitma` | Semicolon-separated list of district identifiers as assigned by the data provider (MITMA). | | `municipalities_mitma` | Semicolon-separated list of municipality identifiers as assigned by the data provider (MITMA). | | `geometry` | Spatial geometry of each district stored as a `MULTIPOLYGON` object, projected in the ETRS89 / UTM zone 30N CRS with XY dimensions. | [^4]: This is likely the population as of end of 2021 or start of 2022. Population for a few districts is missing. Instead of population, residence and overnight stays data may be used as a proxy with caution. Also, newer population figures may be obtained and joined with the provided zones using the reference tables that match the zones ids with official municipal and census district ids from INE. # 2. Mobility data All mobility data is referenced via `id_origin`, `id_destination`, or other location identifiers (mostly labelled as `id`) with the two sets of zones described above. ## 2.1. Origin-destination data {#od-data} The origin-destination data contain the number of trips between `districts`, `municipalities`, or `large urban areas (LUAs)` in Spain for every hour of every day between 2022-02-01 and whichever currently available latest data (2024-06-30 at the time of writing). Each flow also has attributes such as the trip purpose (composed of the type of activity (`home`/`work_or_study`/`frequent_activity`/`infrequent_activity`) at both the origin and destination, but also age, sex, and income of each group of individuals traveling between the origin and destination), province of residence of individuals making this trip, distance covered while making the trip. See the detailed attributes below in a table. Here are the variables you can find in the `district`, `municipality` and `large urban area` level data: | **English Variable Name** | **Original Variable Name** | **Type** | **Description** | |---------|---------|---------|---------------------------------------------| | `date` | `fecha` | `Date` | The date of the recorded data, formatted as `YYYY-MM-DD`. | | `time_slot` | `periodo` | `integer` | The time slot during which the trips occurred. | | `id_origin` | `origen` | `factor` | The origin zone `id` of `district`, `municipality`, or `large urban area`. | | `id_destination` | `destino` | `factor` | The destination zone `id` of `district`, `municipality`, or `large urban area`. | | `distance` | `distancia` | `factor` | The distance range of the trip, categorized into specific intervals such as `0.5-2` (500 m to 2 km), `2-10` (2-10 km), `10-50` (10-50km), and `>50` (50 or more km). | | `activity_origin` | `actividad_origen` | `factor` | The type of activity at the origin zone, recoded from `casa`, `trabajo_estudio`, `frecuente`, `no_frecuente` to `home`, `work_or_study`, `frequent_activity`, `infrequent_activity` respectively. | | `activity_destination` | `actividad_destino` | `factor` | The type of activity at the destination zone, similarly recoded as for `activity_origin` above. | | `study_possible_origin` | `estudio_origen_posible` | `logical` | `TRUE` if the activity at origin may be connected with study, and `FALSE` otherwise. | | `study_possible_destination` | `estudio_destino_posible` | `logical` | `TRUE` if the activity at destination may be connected with study, and `FALSE` otherwise. | | `residence_province_ine_code` | `residencia` | `factor` | The province code of residence of individuals making the trips in `n_trips`, encoded as province codes as classified by the Spanish Statistical Office (INE). | | `residence_province_name` | Derived from `residencia` | `factor` | The full name of the residence province, derived from the province code above. | | `income` | `renta` | `factor` | The income group of individuals making the trips in `n_trips`. Categorized into `<10`, `10-15`, and `>15` (thousands of euros per year). The income for each individual is assigned based on the mean census tract income per person (data source is [INE Household income distribution map](https://www.ine.es/dyngs/INEbase/en/operacion.htm?c=Estadistica_C&cid=1254736177088&menu=ultiDatos&idp=1254735976608)). | | `age` | `edad` | `factor` | The age group of individuals making the trips in `n_trips`. Categorized into `0-25`, `25-45`, `45-65`, `65-100`, or `NA`. The data is partially imputed, for details see [this blogpost](https://www.nommon.es/blog/using-machine-learning-to-predict-sociodemographic-characteristics/) by Nommon. | | `sex` | `sexo` | `factor` | The sex of individuals making the trips in `n_trips`. Categorized into `female`, `male`, or `NA`. The data is partially imputed, for details see [this blogpost](https://www.nommon.es/blog/using-machine-learning-to-predict-sociodemographic-characteristics/) by Nommon. | | `n_trips` | `viajes` | `numeric` | The number of trips for that specific origin-destination pair and time slot. | | `trips_total_length_km` | `viajes_km` | `numeric` | The total length of trips in kilometers, summing up all trips between the origin and destination zones. | | `year` | `year` | `integer` | The year of the recorded data, extracted from the date. | | `month` | `month` | `integer` | The month of the recorded data, extracted from the date. | | `day` | `day` | `integer` | The day of the recorded data, extracted from the date. | **Getting the data** To access the data, use the `spod_get()` function. In this example we will use a short interval of dates: ```{r} dates <- c(start = "2022-01-01", end = "2022-01-04") od_dist <- spod_get(type = "od", zones = "dist", dates = dates) od_muni <- spod_get(type = "od", zones = "muni", dates = dates) ``` The data for the specified dates will be automatically downloaded and cached in the `SPANISH_OD_DATA_DIR` directory. Existing files will not be re-downloaded. **Working with the data** The resulting objects `od_dist` and `od_muni` are of class `tbl_duckdb_connection`[^5]. Basically, you can treat these as regular `data.frame`s or `tibble`s. One important difference is that the data is not actually loaded into memory, because if you requested more dates, e.g. a whole month or a year, all that data would most likely not fit into your computer's memory. A `tbl_duckdb_connection` is mapped to the downloaded CSV files that are cached on disk and the data is only loaded in small chunks as needed at the time of computation. You can manipulate `od_dist` and `od_muni` using `{dplyr}` functions such as `select()`, `filter()`, `mutate()`, `group_by()`, `summarise()`, etc. In the end of any sequence of commands you will need to add `collect()` to execute the whole chain of data manipulations and load the results into memory in an R `data.frame`/`tibble` like so: [^5]: For reference: this object also has classes: `tbl_dbi` ,`tbl_sql`, `tbl_lazy` ,and `tbl` . ```{r} library(dplyr) od_mean_trips_by_ses_over_the_4_days <- od_dist |> group_by(date, age, sex, income) |> summarise( n_trips = sum(n_trips, na.rm = TRUE), .groups = "drop") |> group_by(age, sex, income) |> summarise( daily_mean_n_trips = mean(n_trips, na.rm = TRUE), .groups = "drop") |> collect() od_mean_trips_by_ses_over_the_4_days ``` ``` # A tibble: 39 × 4 age sex income daily_mean_n_trips 1 NA NA <10 7002485. 2 NA NA 10-15 16551405. 3 NA NA >15 2651481. 4 0-25 NA <10 539060. 5 0-25 NA 10-15 1950892. 6 0-25 NA >15 401557. 7 0-25 female <10 1484989. 8 0-25 female 10-15 5357785. 9 0-25 female >15 1764454. 10 0-25 male <10 1558461. # ℹ 29 more rows # ℹ Use `print(n = ...)` to see more rows ``` In this example above, becaus the data is with hourly intervals within each day, we first summed the number of trips for each day by age, sex, and income groups. We then grouped the data again dropping the day variable and calculated the mean number of trips per day by age, sex, and income groups. The full data for all 4 days was probably never loaded into memory all at once. Rather the available memory of the computer was used up to its maximum limit to make that calculation happen, without ever exceeding the available memory limit. If you were doing the same opearation on 100 or even more days, it would work in the same way and would be possible even with limited memory. This is done transparantly to the user with the help of [`DuckDB`](https://duckdb.org/){target="_blank"} (specifically, with [{duckdb} R package](https://r.duckdb.org/index.html){target="_blank"} @duckdb-r). The same summary operation as provided in the example above can be done with the entire dataset for multiple years worth of data on a regular laptop with 8-16 GB memory. It will take a bit of time to complete, but it will be done. To speed things up, please also see the [vignette on converting the data](convert.qmd) into formats that will increase the analsysis performance. {{< include ../inst/vignette-include/csv-date-filter-note.qmd >}} ## 2.2. Number of trips data {#nt-data} For each location, the "number of trips" data provides the number of individuals who spent the night there, with breakdown by the number of trips made, age, and sex. | **English Variable Name** | **Original Variable Name** | **Type** | **Description** | |-----------------|-----------------|-----------------|----------------------| | `date` | `fecha` | `Date` | The date of the recorded data, formatted as `YYYY-MM-DD`. | | `id` | `distrito` | `factor` | The identifier of the `district` or `municipality` zone. | | `age` | `edad` | `factor` | The age group of individuals making the trips in `n_trips`. Categorized into `0-25`, `25-45`, `45-65`, `65-100`, or `NA`. The data is partially imputed, for details see [this blogpost](https://www.nommon.es/blog/using-machine-learning-to-predict-sociodemographic-characteristics/) by Nommon. | | `sex` | `sexo` | `factor` | The sex of individuals making the trips in `n_trips`. Categorized into `female`, `male`, or `NA`. The data is partially imputed, for details see [this blogpost](https://www.nommon.es/blog/using-machine-learning-to-predict-sociodemographic-characteristics/) by Nommon. | | `n_trips` | `numero_viajes` | `factor` | The number of individuals who made trips, categorized by `0`, `1`, `2`, or `2+` trips. | | `n_persons` | `personas` | `factor` | The number of persons making the trips from `district`, `municipality`, or `large urban area (LUA)` with zone `id`. | | `year` | `year` | `integer` | The year of the recorded data, extracted from the date. | | `month` | `month` | `integer` | The month of the recorded data, extracted from the date. | | `day` | `day` | `integer` | The day of the recorded data, extracted from the date. | **Getting the data** To access it use `spod_get()` with `type` set to "number_of_trips", or just "nt". ```{r} dates <- c(start = "2022-01-01", end = "2022-01-04") nt_dist <- spod_get(type = "number_of_trips", zones = "dist", dates = dates) ``` Because this data is small, we can actually load it completely into memory: ```{r} nt_dist_tbl <- nt_dist |> dplyr::collect() ``` ## 2.3. Overnight stays {#os-data} This dataset provides the number of people who spend the night in each location, also identifying their place of residence down to the census district level according to the [INE encoding](https://www.ine.es/ss/Satellite?c=Page&p=1259952026632&pagename=ProductosYServicios%2FPYSLayout&cid=1259952026632&L=1){target="_blank"}. Here are the variables you can find in the `district`, `municipality` and `large urban area` level data: | **English Variable Name** | **Original Variable Name** | **Type** | **Description** | |-----------------|-----------------|-----------------|----------------------| | `date` | `fecha` | `Date` | The date of the recorded data, formatted as `YYYY-MM-DD`. | | `id_residence` | `zona_residencia ` | `factor` | The identifier of the census district according to the [INE encoding](https://www.ine.es/ss/Satellite?c=Page&p=1259952026632&pagename=ProductosYServicios%2FPYSLayout&cid=1259952026632&L=1){target="_blank"}. | | `id_overnight_stay` | `zona_pernoctacion ` | `factor` | The identifier of the `district`, `municipality`, or `large urban area (LUA)` zone. | | `n_persons` | `personas` | `factor` | The number of persons making the trips from `district`, `municipality`, or `large urban area` with zone `id`. | | `year` | `year` | `integer` | The year of the recorded data, extracted from the date. | | `month` | `month` | `integer` | The month of the recorded data, extracted from the date. | | `day` | `day` | `integer` | The day of the recorded data, extracted from the date. | **Getting the data** To access it use `spod_get()` with `type` set to "number_of_trips", or just "nt". ```{r} dates <- c(start = "2022-01-01", end = "2022-01-04") os_dist <- spod_get(type = "overnight_stays", zones = "dist", dates = dates) ``` Because this data is small, we can actually load it completely into memory: ```{r} os_dist_tbl <- os_dist |> dplyr::collect() ```