Skip to content

NASA-GISS/greenland_freshwater

 
 

Repository files navigation

Table of contents

Greenland liquid water discharge from 1950 through 2021

This is the source for “Greenland liquid water discharge from 1958 through 2019” and subsequent versions.

The source for this work is hosted on GitHub at https://github.com/GEUS-PROMICE/freshwater. GitHub issues are used to collect suggested improvements to the paper or problems that made it through review. The work may be under be under active development, including updating data (and therefore tables) within the source document.

Updates since last publication

v 2022-10

v 2022-10 has the following changes:

  • No change to the data values
  • Data product has been reformatted to 4 NetCDF files: One per RCM (2) and domain (2), each containing all years of data
    • E.g., MAR_ice, MAR_land, RACMO_ice, RACMO_Land
  • Data files are now part of the dataset that contains the Geopackages of streams, outlets, and basins, rather than their own dataset on the dataverse
    • DOI: .... has been Deaccesioned
    • DOI: .... now contains the discharge data
  • The README has been updated to show some additional examples using the metadata added in v3

v 2022-08

v 2022-08 has the following changes (see GitHub diff above for more details):

  • Update from BedMachine v3 to v4
  • Data now spans 1950 through 2021, instead of 1958 through 2019
  • Internal NetCDF variable renamed from ‘runoff’ to ‘discharge’
  • Significant improvement in metadata supporting better query by basin, region, or sector
  • Recognition that land runoff with depth << 0 is valid

WARNING

Related Work

Citation

@article{mankoff_2020_liquid,
  author    = {Mankoff, Kenneth D. and Noël, Brice and Fettweis, Xavier and Ahlstrøm, Andreas P. and
                  Colgan, William and Kondo, Ken and Langley, Kirsty and Sugiyama, Shin and van As,
                  Dirk and Fausto, Robert S.},
  title     = {{G}reenland liquid water discharge from 1958 through 2019},
  journal   = {Earth System Science Data},
  year 	    = 2020,
  volume    = 12,
  number    = 4,
  pages     = {2811–2841},
  month     = 11,
  DOI 	    = {10.5194/essd-12-2811-2020},
  publisher = {Copernicus GmbH}
}

Accessing this data

Introduction

NOTE: Data can be accessed directly from the NetCDF files. Querying the NetCDF files directly allows more advanced queries on the metadata, for example, `all outlets with Jakobshavn Isbræ as the nearest discharge gate, excluding outlets more than 5 km away`. The `5 km` filter removes stream discharge from Disko Island which has Jakobshavn Isbræ as the nearest discharge gate, but should not be counted as discharge from that basin.

As an example, it is easiest to begin working with the outlets, save subsetted data, visually check in QGIS, and then when your algorithm appears to work, apply the same query to the discharge NetCDF files. Example:

import pandas as pd
import geopandas as gpd

df = pd.read_csv('./freshwater/ice/outlets.csv', index_col=0)
gdf = gp.GeoDataFrame(df, geometry=gp.points_from_xy(df['lon'],df['lat']))

# select subglacial discharge within 2.5 km of basins
gdf = gdf[(gdf['elev'] < -10) &
          (gdf['M2019_ID_dist'] < 2500)]

gdf.to_file("foo.gpkg", driver="GPKG")

Similar queries might include:

  • Pandas groupby to combine outlets per gate, basin, sector, or region
  • Examining the ice outlet location, and the downstream coastal outlet location. If the two are the same, then the outlet is marine terminating. This may give better results than querying based on the BedMachine provided elev metadata.

If you prefer to not access the NetCDF files directly, after the data have been downloaded the discharge.py script allows access to outlets, basins, and their discharge within a region of interest (ROI). The ROI can be a point, a list describing a polygon, or a file. Optionally, upstream outlets, basins, and discharge from any land outlet(s) can be included. The script can be called from the command line (CLI) or within Python.

The ROI coordinate units can be either EPSG:4326 (lon,lat) or EPSG:3413. The units for the coordinates are guessed using the range of values. If the ROI is a point, basins that contain that point are selected. Either 1 (if the point is on land) or two (ice and the downstream land, if the point is on the ice) basins are selected, and optionally, all ice basins upstream from the one land basin. If the ROI is a polygon, all outlets within the polygon are selected. The polygon does not have to be closed - a convex hull is wrapped around it. If the argument is a file (e.g. KML file) then the first polygon is selected and used.

When the script is run from the command line, CSV data is written to stdout and can be redirected to a file. When the API is accessed from within Python, if the script is used to access outlets, a GeoPandas GeoDataFrame is returned and can be used for further analysis within Python, or written to any file format supported by GeoPandas or Pandas, for example CSV, or GeoPackage for QGIS. If the script is used to access discharge, an xarray Dataset is returned, and can be used for further analysis within Python, or written to any file format supported by xarray, for example CSV or NetCDF.

Database Format

  • The cat column in the CSVs file links to the station vector in the NetCDF.

This script queries two database:

land
The land coast outlets and land basins.
ice
ice margin outlets and ice basins.

The folder structure required is a root folder (named freshwater in the examples below, but can be anything) and then a land and ice sub-folder. The geospatial files for land and ice must be in these folders (i.e. the k=1.0 Streams, Outlets, and Basins dataset from https://dataverse.geus.dk/dataverse/freshwater), along with a MAR.nc and RACMO.nc in each of the land and ice folders.

Example:

./freshwater/ice/
./freshwater/ice/basins.csv
./freshwater/ice/basins_filled.gpkg
./freshwater/ice/basins.gpkg
./freshwater/ice/MAR.nc
./freshwater/ice/outlets.csv
./freshwater/ice/outlets.gpkg
./freshwater/ice/RACMO.nc
./freshwater/ice/streams.csv
./freshwater/ice/streams.gpkg
./freshwater/land/
./freshwater/land/basins.csv
./freshwater/land/basins_filled.gpkg
./freshwater/land/basins.gpkg
./freshwater/land/MAR.nc
./freshwater/land/outlets.csv
./freshwater/land/outlets.gpkg
./freshwater/land/RACMO.nc
./freshwater/land/streams.csv
./freshwater/land/streams.gpkg

Warnings

  • The script takes a few seconds to query the outlets and basins. The script takes ~10s of seconds to query each of the discharge time series datasets. Because there may be up to 6 discharge queries (2 RCMs for each of 1 land domain + ice domain + upstream ice), it can several minutes on a fast laptop to extract the data. To track progress, do not set the quiet flag to True.
  • If a polygon includes ice outlets, and the upstream flag is set, some ice outlets, basins, and discharge may be included twice, once as a “direct” selection within the polygon and once as an upstream outlet and basin from the land polygon. Further processing by the user can remove duplicates (see examples below).
  • The id column may not be unique for multiple reasons:
    • As above, the same outlet may be included twice.
    • id’s are unique within a dataset (i.e. land, and ice), but not between datasets.
  • Due to bash command-line parsing behavior, the syntax --roi -60,60 does not work. Use --roi=-60,06.
  • Longitude is expected in degrees East, and should therefore probably be negative.
  • The cat column in the CSVs file links to the station vector in the NetCDF.
  • If possible, avoid using index-based lookups, and query based on location or station.

Requirements

See environment.yml file in Git repository, or

mamba create -n freshwater_user python=3.7 xarray=0.20.2 fiona=1.8.21 shapely=1.8.2 geopandas=0.7.0 netcdf4=1.6.0 dask=2.15.0
mamba activate freshwater_user

Examples

Command line interface

Usage Instructions

python ./discharge.py -h
usage: discharge.py [-h] --base BASE --roi ROI [-u] (-o | -d) [-q]

Discharge data access

optional arguments:
  -h, --help       show this help message and exit
  --base BASE      Folder containing freshwater data
  --roi ROI        x,y OR lon,lat OR x0,y0 x1,y1 ... xn,yn OR lon0,lat0 lon1,lat1 ... lon_n,lat_n. [lon: degrees E]
  -u, --upstream   Include upstream ice outlets draining into land basins
  -o, --outlets    Return outlet IDs (same as basin IDs)
  -d, --discharge  Return RCM discharge for each domain (outlets merged)
  -q, --quiet      Be quiet

Outlets and basins

One point

The simplest example is a point, in this case near the Watson River outlet. Because we select one point over land and do not request upstream outlets and basins, only one row should be returned.

python ./discharge.py --base ./freshwater --roi=-50.5,67.2 -o -q
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
0112448-51.23367.156-272150-24918504262389117138634ISUNNGUATA-RUSSELLSW195193720Isunnguata Sermia46536landFalse-1-1-1

If we move 10° east to somewhere over the ice, there should be four rows: one for the land outlet and basin, and three more for the three ice scenario:

python ./discharge.py --base ./freshwater --roi=-40.5,67.2 -o -q
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
0118180-38.07166.33313650-2580750-133415796630HELHEIMGLETSCHERSE2319650Helheim Gletsjer11776landFalse-1-1-1
167065-38.1166.333311850-2580650-88414177630HELHEIMGLETSCHERSE2317850Helheim Gletsjer10042iceFalse118180-38.07166.33313650-2580750
Polygon covering multiple land and ice outlets

Here a polygon covers several land outlets near the end of a fjord, and several ice outlets of the nearby ice margin. In addition, we request all ice outlets upstream of all selected land basins.

We use the following simple KML file for our ROI (this can be copied-and-pasted into the Google Earth side-bar to see it). Rather than use this file with --roi=/path/to/file.kml, we use the coordinates directly, and demonstrate dropping the last coordinate because the code will wrap any polygon in a convex hull.

<?xml version="1.0" encoding="UTF-8"?>
<kml xmlns="http://www.opengis.net/kml/2.2" xmlns:gx="http://www.google.com/kml/ext/2.2" xmlns:kml="http://www.opengis.net/kml/2.2" xmlns:atom="http://www.w3.org/2005/Atom">
<Document>
  <name>Ice and Land Sample</name>
  <Placemark>
    <name>ice and land</name>
    <LineString>
      <tessellate>1</tessellate>
      <coordinates>-51.50,66.93 -51.21,66.74 -49.44,66.91 -49.84,67.18 -51.50,66.93</coordinates>
    </LineString>
  </Placemark>
</Document>
</kml>

In this example, we query for upstream outlets, and for brevity show just the first three and last three lines.

python ./discharge.py --base ./freshwater --roi="-51.50,66.93 -51.21,66.74 -49.44,66.91 -49.84,67.18" -q -u -o | (head -n3 ;tail -n4)
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
0113526-50.71367.002-251250-25114501762221847122906ISUNNGUATA-RUSSELLSW195207779Isunnguata Sermia31644landFalse-1-1-1
1113705-50.73566.988-252350-2512850762236837124427ISUNNGUATA-RUSSELLSW195209355Isunnguata Sermia33360landFalse-1-1-1
20067072-49.53866.425-204850-2580850792620400SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262198569Quantum Gletsjer78854iceTrue114921-50.65266.868-250050-2526750
20167096-49.54466.419-205150-258155082562040184SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262197830Quantum Gletsjer78386iceTrue114921-50.65266.868-250050-2526750
20267140-49.53766.407-204950-2582950873620400SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262196481Quantum Gletsjer78243iceTrue114921-50.65266.868-250050-2526750

Discharge

The discharge examples here use the same code as the “outlets and basins” examples above, except we use --discharge rather than --outlet.

One point

The simplest example is a point, in this case near the Watson River outlet. Because we select one point over land and do not request upstream outlets and basins, two time series should be returned: MAR_land and RACMO_land. Rather than showing results for every day from 1958 through 2019, we limit to the header and the first 10 days of June, 2012.

python ./discharge.py --base ./freshwater --roi=-50.5,67.2 -q -d | (head -n1; grep -A9 "^2012-06-01")
timeMAR_landRACMO_land
2012-06-011.2497760.029927
2012-06-020.8310230.001237
2012-06-030.5041990.001330
2012-06-040.5037190.000000
2012-06-050.503333-0.001197
2012-06-060.5107720.304393
2012-06-070.5026830.007438
2012-06-080.5025300.194025
2012-06-092.1959730.087407
2012-06-100.5018200.024703
  • If we move 10° east to somewhere over the ice we add two columns: One for each of the two RCMs over the ice domain.
  • If the --upstream flag is set, we add two columns: One for each of the RCMs over the upstream ice domains. Results are summed across outlets per domain.
  • Results are therefore one of the following
    • Two columns: 2 RCM * 1 land domain
    • Four columns: 2 RCM * (1 land + 1 ice domain)
    • Four columns: 2 RCM * (1 land + 1 upstream ice domain)
    • Six columns: 2 RCM * (1 land + 1 ice + 1 upstream ice domain)
Polygon covering multiple land and ice outlets

When querying using an ROI that covers multiple outlets, discharge is summed by domain. Therefore, even if 100s of outlets are within the ROI, either two columns, eight, eight, or fourteen columns are returned depending on the options.

Python API

The python API is similar to the command line interface, but rather than printing results to stdout, returns a GeoPandas GeoDataFrame of outlets, an xarray Dataset of discharge. The discharge is not summed by domain, but instead contains discharge for each outlet.

Outlets and basins

One point

The simplest example is a point, in this case near the Watson River outlet. Because we select one point over land and do not request upstream outlets and basins, only one row should be returned.

from discharge import discharge 
df = discharge(base="./freshwater", roi="-50.5,67.2", quiet=True).outlets()

The df variable is a Pandas GeoDataFrame.

It includes two geometry columns

outlet
A point for the location of the outlet (also available as the x and y columns)
basin
A polygon describing this basin

Because the geometry columns do not display well in tabular form, we drop them.

df.drop(columns=["outlet","basin"])
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
0112448-51.232967.1555-272150-24918504262389117138634ISUNNGUATA-RUSSELLSW195193720Isunnguata Sermia46536landFalse-1nannan-1-1
Polygon covering multiple land and ice outlets

Here a polygon covers several land outlets near the end of a fjord, and several ice outlets of the nearby ice margin. In addition, we request all ice outlets upstream of all selected land basins. Results are shown in tabular form and written to geospatial file formats.

from discharge import discharge
df = discharge(base="./freshwater", roi="-51.50,66.93 -51.21,66.74 -49.44,66.91 -49.84,67.18", quiet=True, upstream=True).outlets()

View the first few rows, excluding the geometry columns

df.drop(columns=["outlet","basin"]).head()
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
0113526-50.71367.0017-251250-25114501762221847122906ISUNNGUATA-RUSSELLSW195207779Isunnguata Sermia31644landFalse-1nannan-1-1
1113705-50.734666.9884-252350-2512850762236837124427ISUNNGUATA-RUSSELLSW195209355Isunnguata Sermia33360landFalse-1nannan-1-1
2113729-50.777166.9849-254250-2513050-162255387126272ISUNNGUATA-RUSSELLSW195209906Isunnguata Sermia35015landFalse-1nannan-1-1
3113767-50.863466.9752-258150-25137501462294577130178ISUNNGUATA-RUSSELLSW195211369Isunnguata Sermia38643landFalse-1nannan-1-1
4113787-50.957566.9688-262350-25140501162335597134261ISUNNGUATA-RUSSELLSW195212574Isunnguata Sermia42388landFalse-1nannan-1-1

View the last few rows:

Note that the domain and upstream columns can be used to subset the table.

df.drop(columns=["outlet","basin"]).tail()
indexidlonlatxyelevZ2012_sectorZ2012_sector_distM2019_IDM2019_ID_distM2019_basinM2019_regionM2020_gateM2020_gate_distB2015_nameB2015_distdomainupstreamcoast_idcoast_loncoast_latcoast_xcoast_y
19867008-49.538666.4387-204750-25793507596204052SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262200066Quantum Gletsjer79350iceTrue114921-50.651766.8677-250050-2526750
19967022-49.520666.4375-203950-2579550754620400SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262199999Quantum Gletsjer80065iceTrue114921-50.651766.8677-250050-2526750
20067072-49.538266.4254-204850-2580850792620400SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262198569Quantum Gletsjer78854iceTrue114921-50.651766.8677-250050-2526750
20167096-49.543666.419-205150-258155082562040184SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262197830Quantum Gletsjer78386iceTrue114921-50.651766.8677-250050-2526750
20267140-49.536866.4068-204950-2582950873620400SAQQAP-MAJORQAQ-SOUTHTERRUSSEL_SOUTHQUARUSSELSW262196481Quantum Gletsjer78243iceTrue114921-50.651766.8677-250050-2526750

Finally, write data to various file formats. GeoPandas DataFrames can only have one geometry, so we must select one and drop the other before writing the file.

df.drop(columns=["outlet","basin"]).to_csv("outlets.csv")
df.set_geometry("outlet").drop(columns="basin").to_file("outlets.gpkg", driver="GPKG")
df.set_geometry("basin").drop(columns="outlet").to_file("basins.gpkg", driver="GPKG")

Discharge

The code here is the same as above from the “Outlets and basins” section, but we call discharge() rather than outlets().

One point

The simplest example is a point, in this case near the Watson River outlet. Because we select one point over land and do not request upstream outlets and basins, only one row should be returned.

from discharge import discharge
ds = discharge(base="./freshwater", roi="-50.5,67.2").discharge()

Print the xarray Dataset:

print(ds)
<xarray.Dataset>
Dimensions:     (land: 1, time: 26298)
Coordinates:
  * time        (time) datetime64[ns] 1950-01-01 1950-01-02 ... 2021-12-31
  * land        (land) uint64 112448
Data variables:
    MAR_land    (time, land) float64 0.03053 0.03059 0.03063 ... nan nan nan
    RACMO_land  (time, land) float64 nan nan nan nan ... 0.04167 0.03844 0.03304

Display the time series. Unlike the command line interface, here the outlets are not merged.

ds.sel(time=slice('2012-06-01','2012-06-10')).to_dataframe()
MAR_landRACMO_land
(112448, Timestamp(‘2012-06-01 00:00:00’))1.249780.0299266
(112448, Timestamp(‘2012-06-02 00:00:00’))0.8310230.00123663
(112448, Timestamp(‘2012-06-03 00:00:00’))0.5041990.00133009
(112448, Timestamp(‘2012-06-04 00:00:00’))0.5037190
(112448, Timestamp(‘2012-06-05 00:00:00’))0.503333-0.00119686
(112448, Timestamp(‘2012-06-06 00:00:00’))0.5107720.304393
(112448, Timestamp(‘2012-06-07 00:00:00’))0.5026830.00743797
(112448, Timestamp(‘2012-06-08 00:00:00’))0.502530.194025
(112448, Timestamp(‘2012-06-09 00:00:00’))2.195970.0874073
(112448, Timestamp(‘2012-06-10 00:00:00’))0.501820.0247026

In order to merge the outlets, select all coordinates that are not time and merge them. Also, apply a rolling mean:

dims = [_ for _ in ds.dims.keys() if _ != 'time']  # get all dimensions except the time dimension
ds.sum(dim=dims)\
  .rolling(time=7)\
  .mean()\
  .sel(time=slice('2012-06-01','2012-06-10'))\
  .to_dataframe()
timeMAR_landRACMO_land
2012-06-01 00:00:008.607731.39995
2012-06-02 00:00:008.358131.24619
2012-06-03 00:00:005.533620.460494
2012-06-04 00:00:003.021510.15819
2012-06-05 00:00:001.454690.089366
2012-06-06 00:00:000.7735390.0882229
2012-06-07 00:00:000.6579290.0490182
2012-06-08 00:00:000.551180.0724609
2012-06-09 00:00:000.7461730.084771
2012-06-10 00:00:000.7458330.08811
Polygon covering multiple land and ice outlets

Here a polygon covers several land outlets near the end of a fjord, and several ice outlets of the nearby ice margin. In addition, we request all ice outlets upstream of all selected land basins.

from discharge import discharge
ds = discharge(base="./freshwater", roi="-51.50,66.93 -51.21,66.74 -49.44,66.91 -49.84,67.18", quiet=True, upstream=True).discharge()

What are the dimensions (i.e. how many outlets in each domain?)

print(ds)
<xarray.Dataset>
Dimensions:             (ice: 35, ice_upstream: 84, land: 84, time: 26298)
Coordinates:
  * ice_upstream        (ice_upstream) uint64 65473 65477 65483 ... 67096 67140
  * time                (time) datetime64[ns] 1950-01-01 ... 2021-12-31
  * land                (land) uint64 113526 113705 113729 ... 115311 115336
  * ice                 (ice) uint64 65487 65492 65509 ... 65668 65671 65714
Data variables:
    MAR_land            (time, land) float64 0.005423 1.435e-05 ... nan nan
    MAR_ice             (time, ice) float64 3.245e-15 1.975e-16 ... nan nan
    RACMO_land          (time, land) float64 nan nan nan ... 0.001391 0.02869
    RACMO_ice           (time, ice) float64 nan nan nan ... 0.006565 0.003061
    MAR_ice_upstream    (time, ice_upstream) float64 2.406e-16 3.296e-16 ... nan
    RACMO_ice_upstream  (time, ice_upstream) float64 nan nan ... 4.135e-05

With these results:

  • Sum all outlets within each domain
  • Drop the land discharge and the upstream domains (keep only ice discharge explicitly within our ROI)
  • Apply a 5-day rolling mean
  • Plot 2012 discharge season
d = [_ for _ in ds.dims.keys() if _ != 'time'] # dims for summing (don't sum time dimension)
v = [_ for _ in ds.data_vars if ('land' in _) | ('_u' in _)] # vars containing '_u'

r = ds.sum(dim=d)\
      .drop_vars(v)\
      .rolling(time=5).mean()

import matplotlib.pyplot as plt
plt.style.use('seaborn')

for d in r.data_vars: r[d].sel(time=slice('2012-04-01','2012-11-15')).plot(drawstyle='steps', label=d)
_ = plt.legend()
plt.savefig("./fig/api_example.png", bbox_inches='tight')

./fig/api_example.png

About

Greenland liquid water runoff from 1958 through last year

Resources

Stars

Watchers

Forks

Packages

No packages published

Languages

  • TeX 93.5%
  • Python 6.5%