Skip to content

Commit 874b6a4

Browse files
Remote datasets: Add "load_earth_mean_dynamic_topography" to load "CNES Earth Mean Dynamic Topography" dataset (#3718)
Co-authored-by: Dongdong Tian <[email protected]>
1 parent 41702d7 commit 874b6a4

6 files changed

+175
-0
lines changed

doc/api/index.rst

+1
Original file line numberDiff line numberDiff line change
@@ -238,6 +238,7 @@ and store them in GMT's user data directory.
238238
datasets.load_earth_geoid
239239
datasets.load_earth_magnetic_anomaly
240240
datasets.load_earth_mask
241+
datasets.load_earth_mean_dynamic_topography
241242
datasets.load_earth_relief
242243
datasets.load_earth_vertical_gravity_gradient
243244
datasets.load_mars_relief

pygmt/datasets/__init__.py

+3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@
1111
from pygmt.datasets.earth_geoid import load_earth_geoid
1212
from pygmt.datasets.earth_magnetic_anomaly import load_earth_magnetic_anomaly
1313
from pygmt.datasets.earth_mask import load_earth_mask
14+
from pygmt.datasets.earth_mean_dynamic_topography import (
15+
load_earth_mean_dynamic_topography,
16+
)
1417
from pygmt.datasets.earth_night import load_black_marble
1518
from pygmt.datasets.earth_relief import load_earth_relief
1619
from pygmt.datasets.earth_vertical_gravity_gradient import (
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,103 @@
1+
"""
2+
Function to download the CNES Earth mean dynamic topography dataset from the GMT data
3+
server, and load as :class:`xarray.DataArray`.
4+
5+
The grids are available in various resolutions.
6+
"""
7+
8+
from collections.abc import Sequence
9+
from typing import Literal
10+
11+
import xarray as xr
12+
from pygmt.datasets.load_remote_dataset import _load_remote_dataset
13+
14+
__doctest_skip__ = ["load_earth_mean_dynamic_topography"]
15+
16+
17+
def load_earth_mean_dynamic_topography(
18+
resolution: Literal["01d", "30m", "20m", "15m", "10m", "07m"] = "01d",
19+
region: Sequence[float] | str | None = None,
20+
registration: Literal["gridline", "pixel"] = "gridline",
21+
) -> xr.DataArray:
22+
r"""
23+
Load the CNES Earth mean dynamic topography dataset in various resolutions.
24+
25+
.. figure:: https://www.generic-mapping-tools.org/remote-datasets/_images/GMT_earth_mdt.jpg
26+
:width: 80 %
27+
:align: center
28+
29+
CNES Earth mean dynamic topography dataset.
30+
31+
The grids are downloaded to a user data directory (usually
32+
``~/.gmt/server/earth/earth_mdt/``) the first time you invoke this function.
33+
Afterwards, it will load the grid from the data directory. So you'll need an
34+
internet connection the first time around.
35+
36+
These grids can also be accessed by passing in the file name
37+
**@earth_mdt**\_\ *res*\[_\ *reg*] to any grid processing function or plotting
38+
method. *res* is the grid resolution (see below), and *reg* is the grid registration
39+
type (**p** for pixel registration or **g** for gridline registration).
40+
41+
The default color palette table (CPT) for this dataset is *@earth_mdt.cpt*. It's
42+
implicitly used when passing in the file name of the dataset to any grid plotting
43+
method if no CPT is explicitly specified. When the dataset is loaded and plotted
44+
as an :class:`xarray.DataArray` object, the default CPT is ignored, and GMT's
45+
default CPT (*turbo*) is used. To use the dataset-specific CPT, you need to
46+
explicitly set ``cmap="@earth_mdt.cpt"``.
47+
48+
Refer to :gmt-datasets:`earth-mdt.html` for more details about available datasets,
49+
including version information and references.
50+
51+
Parameters
52+
----------
53+
resolution
54+
The grid resolution. The suffix ``d`` and ``m`` stand for arc-degrees and
55+
arc-minutes. Note that ``"07m"`` refers to a resolution of 7.5 arc-minutes.
56+
region
57+
The subregion of the grid to load, in the form of a sequence [*xmin*, *xmax*,
58+
*ymin*, *ymax*] or an ISO country code.
59+
registration
60+
Grid registration type. Either ``"pixel"`` for pixel registration or
61+
``"gridline"`` for gridline registration.
62+
63+
Returns
64+
-------
65+
grid
66+
The CNES Earth mean dynamic topography grid. Coordinates are latitude and
67+
longitude in degrees. Values are in meters.
68+
69+
Note
70+
----
71+
The registration and coordinate system type of the returned
72+
:class:`xarray.DataArray` grid can be accessed via the GMT accessors (i.e.,
73+
``grid.gmt.registration`` and ``grid.gmt.gtype`` respectively). However, these
74+
properties may be lost after specific grid operations (such as slicing) and will
75+
need to be manually set before passing the grid to any PyGMT data processing or
76+
plotting functions. Refer to :class:`pygmt.GMTDataArrayAccessor` for detailed
77+
explanations and workarounds.
78+
79+
Examples
80+
--------
81+
82+
>>> from pygmt.datasets import load_earth_mean_dynamic_topography
83+
>>> # load the default grid (gridline-registered 1 arc-degree grid)
84+
>>> grid = load_earth_mean_dynamic_topography()
85+
>>> # load the 30 arc-minutes grid with "gridline" registration
86+
>>> grid = load_earth_mean_dynamic_topography(
87+
resolution="30m", registration="gridline"
88+
...)
89+
>>> # load high-resolution (5 arc-minutes) grid for a specific region
90+
>>> grid = load_earth_mean_dynamic_topography(
91+
... resolution="05m",
92+
... region=[120, 160, 30, 60],
93+
... registration="gridline",
94+
... )
95+
"""
96+
grid = _load_remote_dataset(
97+
name="earth_mdt",
98+
prefix="earth_mdt",
99+
resolution=resolution,
100+
region=region,
101+
registration=registration,
102+
)
103+
return grid

pygmt/datasets/load_remote_dataset.py

+13
Original file line numberDiff line numberDiff line change
@@ -246,6 +246,19 @@ class GMTRemoteDataset(NamedTuple):
246246
"30s": Resolution("30s", registrations=["pixel"]),
247247
},
248248
),
249+
"earth_mdt": GMTRemoteDataset(
250+
description="CNES Earth mean dynamic topography",
251+
units="meters",
252+
extra_attributes={"horizontal_datum": "WGS84"},
253+
resolutions={
254+
"01d": Resolution("01d"),
255+
"30m": Resolution("30m"),
256+
"20m": Resolution("20m"),
257+
"15m": Resolution("15m"),
258+
"10m": Resolution("10m"),
259+
"07m": Resolution("07m", registrations=["gridline"]),
260+
},
261+
),
249262
"earth_vgg": GMTRemoteDataset(
250263
description="IGPP Earth vertical gravity gradient",
251264
units="Eotvos",

pygmt/helpers/caching.py

+2
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ def cache_data():
2323
"@earth_mag_01d_g",
2424
"@earth_mag4km_01d_g",
2525
"@earth_mask_01d_g",
26+
"@earth_mdt_01d_g",
27+
"@earth_mdt_07m_g",
2628
"@earth_night_01d",
2729
"@earth_relief_01d_g",
2830
"@earth_relief_01d_p",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
"""
2+
Test basic functionality for loading Earth mean dynamic topography datasets.
3+
"""
4+
5+
import numpy as np
6+
import numpy.testing as npt
7+
from pygmt.datasets import load_earth_mean_dynamic_topography
8+
9+
10+
def test_earth_mdt_01d():
11+
"""
12+
Test some properties of the Earth mean dynamic topography 01d data.
13+
"""
14+
data = load_earth_mean_dynamic_topography(resolution="01d")
15+
assert data.name == "z"
16+
assert data.attrs["description"] == "CNES Earth mean dynamic topography"
17+
assert data.attrs["units"] == "meters"
18+
assert data.attrs["horizontal_datum"] == "WGS84"
19+
assert data.shape == (181, 361)
20+
assert data.gmt.registration == 0
21+
npt.assert_allclose(data.lat, np.arange(-90, 91, 1))
22+
npt.assert_allclose(data.lon, np.arange(-180, 181, 1))
23+
npt.assert_allclose(data.min(), -1.4668, atol=0.0001)
24+
npt.assert_allclose(data.max(), 1.7151, atol=0.0001)
25+
26+
27+
def test_earth_mdt_01d_with_region():
28+
"""
29+
Test loading low-resolution Earth mean dynamic topography with "region".
30+
"""
31+
data = load_earth_mean_dynamic_topography(resolution="01d", region=[-10, 10, -5, 5])
32+
assert data.shape == (11, 21)
33+
assert data.gmt.registration == 0
34+
npt.assert_allclose(data.lat, np.arange(-5, 6, 1))
35+
npt.assert_allclose(data.lon, np.arange(-10, 11, 1))
36+
npt.assert_allclose(data.min(), 0.346, atol=0.0001)
37+
npt.assert_allclose(data.max(), 0.4839, atol=0.0001)
38+
39+
40+
def test_earth_mdt_07m_default_registration():
41+
"""
42+
Test that the grid returned by default for the 7 arc-minutes resolution has a
43+
"gridline" registration.
44+
"""
45+
data = load_earth_mean_dynamic_topography(resolution="07m", region=[-10, -9, 3, 5])
46+
assert data.shape == (17, 9)
47+
assert data.gmt.registration == 0
48+
assert data.coords["lat"].data.min() == 3.0
49+
assert data.coords["lat"].data.max() == 5.0
50+
assert data.coords["lon"].data.min() == -10.0
51+
assert data.coords["lon"].data.max() == -9.0
52+
npt.assert_allclose(data.min(), 0.4138, atol=0.0001)
53+
npt.assert_allclose(data.max(), 0.4302, atol=0.0001)

0 commit comments

Comments
 (0)