Reprojecting spatial data is a common task in GIS. Coordinate reference system (:term:`CRS`) transformations are handled by |PROJ| - a required dependency of GDAL. In this tutorial we'll walk through reprojecting both vector and raster data using data from the Natural Earth dataset.
We'll start by getting projection information from one of the Natural Earth rasters.
gdal raster info NE2_50M_SR.tifBy default, :ref:`gdal_raster_info` returns a summary of the CRS in a human-readable format. The CRS information in the output is shown below:
Coordinate Reference System: - name: WGS 84 - ID: EPSG:4326 - type: Geographic 2D - area of use: World, west -180.00, south -90.00, east 180.00, north 90.00 Data axis to CRS axis mapping: 2,1
These key details are explained below:
- name: WGS 84 - the name of the CRS. This is a human-readable name, and is not guaranteed to be unique.
- ID: EPSG:4326 - the unique identifier for the CRS in the :term:`EPSG` registry.
- type: Geographic 2D - this indicates that the CRS is a geographic coordinate system (latitude and longitude), rather than a projected coordinate system (e.g. Web Mercator, UTM). The "2D" indicates that the CRS has two dimensions (latitude and longitude), as opposed to a 3D CRS which would include a vertical component (e.g. height).
- area of use: World, west -180.00, south -90.00, east 180.00, north 90.00 - this indicates the area of the world where the CRS is intended to be used. In this case, it covers the entire world.
- Data axis to CRS axis mapping: 2,1 - this is a GDAL specific line showing how raster data axes map to the CRS axes. Here, the raster uses (longitude, latitude) order, while the CRS defines axes as (latitude, longitude), so GDAL maps them accordingly.
We can get more detailed information about the CRS by providing the --crs-format option.
gdal raster info NE2_50M_SR.tif --crs-format=WKT2This shows the definition of the CRS in the WKT2 (Well-Known Text version 2). This format was defined by the |OGC| and represents a complete CRS definition, not just a projection. It includes everything GDAL needs to perform coordinate transformations - a full datum definition, the projection method and parameters (for projected CRSs), and axis definitions. The full specification is found at http://www.opengis.net/doc/is/crs-wkt/2.1.11, and the current widely adopted version is ISO 19162:2019, which GDAL supports.
A breakdown of the WKT2 CRS contents is provided below.
Coordinate System is:
GEOGCRS["WGS 84",
ENSEMBLE["World Geodetic System 1984 ensemble",
MEMBER["World Geodetic System 1984 (Transit)"],
MEMBER["World Geodetic System 1984 (G730)"],
MEMBER["World Geodetic System 1984 (G873)"],
MEMBER["World Geodetic System 1984 (G1150)"],
MEMBER["World Geodetic System 1984 (G1674)"],
MEMBER["World Geodetic System 1984 (G1762)"],
MEMBER["World Geodetic System 1984 (G2139)"],
MEMBER["World Geodetic System 1984 (G2296)"],
ELLIPSOID["WGS 84",6378137,298.257223563,
LENGTHUNIT["metre",1]],
ENSEMBLEACCURACY[2.0]],
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]],
USAGE[
SCOPE["Horizontal component of 3D system."],
AREA["World."],
BBOX[-90,-180,90,180]],
ID["EPSG",4326]]
Data axis to CRS axis mapping: 2,1GEOGCRS["WGS 84"This means it's a Geographic CRS (latitude and longitude, not a projected CRS like Web Mercator or UTM), and has the name "WGS 84".ENSEMBLE["World Geodetic System 1984 ensemble",..- WGS 84 has been updated multiple times. The ENSEMBLE list contains all the different versions.ENSEMBLEACCURACY[2.0]means all the different versions are considered equivalent within ~2 metres.ELLIPSOID["WGS 84", 6378137, 298.257223563]defines the Earth's shape.PRIMEM["Greenwich",0...states the Prime meridian is at Greenwich (0° longitude).CS[ellipsoidal,2],the Coordinate System is a 2-dimensional ellipsoidal.AXIS["geodetic longitude (Lon)",east, ORDER[2],details the axis order. In this case the CRS defines axis order as latitude, longitude. This differs from the traditional GIS display order (longitude, latitude), which can lead to confusion - for more discussion around this see :ref:`osr_api_tut_axis_order`.USAGE[...contains metadata associated with the coordinate system.ID["EPSG",4326]- this is the familiar EPSG:4326 code used to reference the CRS. This identifier links the WKT definition to the EPSG registry, but the WKT provides a fully expanded CRS definition that GDAL can use directly for coordinate transformations.
You can look up the details for each coordinate system at https://spatialreference.org, for example https://spatialreference.org/ref/epsg/4326/. These pages also show coordinate systems in other formats. Alternatively, GDAL includes :ref:`gdalsrsinfo`, a command-line tool to display and validate a CRS:
gdalsrsinfo -V EPSG:27561GDAL can work with the many different CRS formats supported by PROJ, see :example:`gdal-vector-reproject-crs`. See the full list of options at the projinfo page.
You can also use --output-crs=PROJJSON to return the CRS definition in PROJJSON format, which is a JSON-based format defined by PROJ for representing coordinate reference systems.
This is generally easier to parse programmatically than WKT, and is also the format used when outputting dataset details in JSON (e.g. with --format=JSON).
See :example:`gdal-raster-info-crs` for an example of parsing CRS information from JSON output.
gdal raster info NE2_50M_SR.tif --crs-format=PROJJSON
gdal raster info NE2_50M_SR.tif --format=JSONNow let's reproject the raster dataset to Web Mercator, and look at the CRS information.
gdal raster reproject --dst-crs=EPSG:3857 NE2_50M_SR.tif NE2_50M_SR_3857.tif
gdal raster info NE2_50M_SR_3857.tifNote
If you are using
bashyou can take advantage of auto-completion for CRS codes, see :ref:`gdal_bash_completion`. This suggests potential CRS codes, but taking advantage of this requires the input dataset to be specified first:$ gdal raster reproject NE2_50M_SR.tif --dst-crs=EPSG:385<TAB> 3850 -- ETRS89-SWE [SWEREF 99] / RT90 5 gon O emulation 3852 -- RSRGD2000 / DGLC2000 3855 -- EGM2008 height 3851 -- NZGD2000 / NZCS2000 3854 -- County ST74 3857 -- WGS 84 / Pseudo-Mercator
As this example uses a global raster, many CRS options are shown. When working with a more localized dataset, auto-completion will suggest only CRSs relevant to the area covered by the data.
The CRS information from the above command is shown below.
Coordinate Reference System: - name: WGS 84 / Pseudo-Mercator - ID: EPSG:3857 - type: Projected - projection type: Popular Visualisation Pseudo Mercator - units: metre - area of use: World between 85.06°S and 85.06°N, west -180.00, south -85.06, east 180.00, north 85.06 Data axis to CRS axis mapping: 1,2
The key differences to note here are:
- type: Projected - this indicates that the CRS is a projected coordinate system, which uses a map projection to convert geographic coordinates (latitude and longitude) into planar coordinates (easting and northing).
- projection type: Popular Visualisation Pseudo Mercator - this shows both the projection conversion name and method name. In this case the names are identical, but in some cases they may differ (e.g. the conversion name may be "UTM zone 33N" while the method is "Transverse Mercator").
- units: metre - the units of the projected coordinates are in metres, rather than degrees as in the original geographic CRS.
- Data axis to CRS axis mapping: 1,2 - this indicates that the data axis order matches the CRS axis order (X, Y). In contrast, the original geographic CRS (EPSG:4326) defines axes as (latitude, longitude), requiring a mapping of 2,1.
Let's also look at the full WKT definition for this CRS.
gdal raster info NE2_50M_SR_3857.tif --crs-format=WKT2The CRS output is shown below.
Layer SRS WKT:
PROJCRS["WGS 84 / Pseudo-Mercator",
BASEGEOGCRS["WGS 84",
ENSEMBLE["World Geodetic System 1984 ensemble",
MEMBER["World Geodetic System 1984 (Transit)"],
MEMBER["World Geodetic System 1984 (G730)"],
MEMBER["World Geodetic System 1984 (G873)"],
MEMBER["World Geodetic System 1984 (G1150)"],
MEMBER["World Geodetic System 1984 (G1674)"],
MEMBER["World Geodetic System 1984 (G1762)"],
MEMBER["World Geodetic System 1984 (G2139)"],
MEMBER["World Geodetic System 1984 (G2296)"],
ELLIPSOID["WGS 84",6378137,298.257223563,
LENGTHUNIT["metre",1]],
ENSEMBLEACCURACY[2.0]],
PRIMEM["Greenwich",0,
ANGLEUNIT["degree",0.0174532925199433]],
ID["EPSG",4326]],
CONVERSION["Popular Visualisation Pseudo-Mercator",
METHOD["Popular Visualisation Pseudo Mercator",
ID["EPSG",1024]],
PARAMETER["Latitude of natural origin",0,
ANGLEUNIT["degree",0.0174532925199433],
ID["EPSG",8801]],
PARAMETER["Longitude of natural origin",0,
ANGLEUNIT["degree",0.0174532925199433],
ID["EPSG",8802]],
PARAMETER["False easting",0,
LENGTHUNIT["metre",1],
ID["EPSG",8806]],
PARAMETER["False northing",0,
LENGTHUNIT["metre",1],
ID["EPSG",8807]]],
CS[Cartesian,2],
AXIS["easting (X)",east,
ORDER[1],
LENGTHUNIT["metre",1]],
AXIS["northing (Y)",north,
ORDER[2],
LENGTHUNIT["metre",1]],
USAGE[
SCOPE["Web mapping and visualisation."],
AREA["World between 85.06°S and 85.06°N."],
BBOX[-85.06,-180,85.06,180]],
ID["EPSG",3857]]
Data axis to CRS axis mapping: 1,2PROJCRS["WGS 84 / Pseudo-Mercator",- this is a projected CRS rather than a geographic CRS.BASEGEOGCRS["WGS 84", ...]- the underlying coordinate reference system is based on WGS 84. The same properties as above are listed here.CONVERSION["Popular Visualisation Pseudo-Mercator", ...]this details the map projection itself.CS[Cartesian,2]- the projected units are a 2D Cartesian system (X, Y).AXIS["easting (X)",east] AXIS["northing (Y)",north]- X = easting, Y = northing.LENGTHUNIT["metre",1]- units are in metres.USAGE[- the metadata indicating the CRS is intended for web mapping, and also that theBBOXdoes not cover the full extent of WGS84 and cannot represent data at the poles.
In practice, you will often refer to a CRS using an EPSG code (e.g. EPSG:4326), but GDAL internally works with full WKT definitions like the one shown above. Next, we'll use this CRS information to reproject both raster and vector datasets.
Reprojecting raster data is handled by :ref:`gdal_raster_reproject` (see this page for further examples). Let's try an example using Natural Earth raster data. The original data is in WGS 84 (EPSG:4326) and we'll reproject it to a CRS defined outside the EPSG registry, ESRI:53009 - the Mollweide projection. As this is a large file we'll also resize it to 10% of the original size to speed up the reprojection. The order of operations in the pipeline matters, and in this case we want to resize before reprojecting - this changes the processing time from a few minutes to a few seconds, but also reduces the output resolution and may affect reprojection accuracy.
.. tabs::
.. code-tab:: bash
gdal raster pipeline \
! read NE2_50M_SR.tif \
! resize --size=10%,10% -r average \
! reproject --dst-crs ESRI:53009 \
! write NE2_50M_SR_53009.png --overwrite
.. code-tab:: ps1
gdal raster pipeline `
! read NE2_50M_SR.tif `
! resize --size=10%,10% -r average `
! reproject --dst-crs ESRI:53009 `
! write NE2_50M_SR_53009.png --overwrite
Note that the output is a PNG file, and the CRS information is stored in a companion file with the same name, using the extension .aux.xml.
GDAL refers to these as :term:`PAM` (Persistent Auxiliary Metadata) files. They are used to store metadata that cannot
be stored in the file itself, such as the CRS for formats that do not support it (e.g. PNG, JPEG).
Reprojecting vector data is handled by :ref:`gdal_vector_reproject`. We can reproject the entire Natural Earth vector dataset to Web Mercator with the following command:
gdal vector reproject --dst-crs=EPSG:3857 natural_earth_vector.gpkg natural_earth_vector_3857.gpkg --overwriteIf you want to reproject just a single layer, you can specify the layer name with the --layer option. For example, to reproject just the ne_10m_admin_0_countries layer:
gdal vector reproject --dst-crs=EPSG:3857 --layer=ne_10m_admin_0_countries natural_earth_vector.gpkg natural_earth_vector_3857.gpkg --overwriteIf your data is already in the correct CRS, but is missing metadata, you can use :ref:`gdal_vector_edit` to assign the CRS information without reprojecting the data. For example, if you have a GeoPackage that is in WGS 84 but is missing the CRS metadata, you can add it with the following command:
gdal vector edit --crs=EPSG:4326 in.gpkg out.gpkg --overwrite- For working with projections using the C++ API, see :ref:`osr_api_tut`.
- For discussions about WKT and GDAL, see :ref:`wktproblems`.
