Skip to content

Basic Usage

This page gives some common basic usage examples. For more detailed information, see the documentation for the SDKs and the REST reference.

Setup

If you have an existing Python environment set up, you can add fathom-global-client-sdk to your existing requirements.txt file and install as normal.

To create a new virtualenv to use the Fathom Python module, proceed with the following steps.

# ensure you have the right requirements
python3 -m ensurepip

# Go to your project directory
cd /path/to/working/dir/
# Create a virtualenv in the 'venv' folder and activate it
python3 -m venv venv
source venv/bin/activate

# Install the package from pypi into the virtualenv
pip3 install fathom-global-client-sdk

The suggested way to interact with the Fathom API on the command line is using curl.

Authentication

Authentication tokens are required for API requests.

The SDK performs the obtaining of tokens for the user when creating a Client object. Just use the access credentials supplied by Fathom.

from fathom.sdk.v2 import Client

client = Client(
    "client_id",
    "client_secret",
)

# Future requests will be authenticated

To access data from the REST API endpoints you will need to authorize requests using a JWT token in the Authorization header. To be able to do this, you must obtain a token from our authorization server using your supplied client_id and client_secret.

curl --request POST \
  --url https://api.fathom.global/v2/auth/token \
  --header 'content-type: application/json' \
  --data '{"client_id":"CLIENT_ID","client_secret":"CLIENT_SECRET"}'

You should obtain a response like:

{
  "access_token": "access_token",
  "expire_secs": 86400
}

When you have the Auth0 JWT token you will need to pass it as a header with every request to any of the Fathom API endpoints using the pattern:

Authorization: Bearer ACCESS_TOKEN

Data queries

Key Values

There are some integer values that are used to represent special cases in the data. These are:

Value Title Examples
-32768 No Data A point in the ocean or an inland point on a coastal layer
-32767 Permanent Water A point just offshore or in a river

Project ID metadata

In some cases you may require Fathom API queries to be billed as part of an internal project. The SDK allows you to attach a project ID as part of a query. These project IDs are stored in Fathom's usage database and will be used to itemise usage for billing purposes.

Methods to fetch data take an optional project_id keyword argument that will pass the project ID along:

project_id = "large-engineering-project"
client.geo.get_points(([pt]), layer_ids, project_id)

Add the project id in a "metadata" dictionary in the request body:

curl --location --request POST 'https://api.fathom.global/v2/points:data' \
--header 'Content-Type: application/json' \
--header 'Authorization: Bearer ACCESS_TOKEN' \
--data-raw '{
    "metadata": {"project_id": "large-engineering-project"},
    "points": [
        ...
    ],
    "layer_ids": {
        ...
    }
  }'

Point queries

To obtain values at specified points:

pts = [
    point(lat=51.448765, lng=-2.601340),
    point(lat=39.628574, lng=-105.1162333)
]

layer_ids = [
    "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0",
    "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0",
    "GLOBAL-1ARCSEC-NW_OFFSET-1in100-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0"
]

data = client.geo.get_points(pts, layer_ids)

To obtain values at specified points:

Request:

curl --location --request POST 'https://api.fathom.global/v2/points:data' \
  --header 'Content-Type: application/json' \
  --header 'Authorization: Bearer ACCESS_TOKEN' \
  --data-raw '{
    "points": [
      {
        "longitude": -2.601340,
        "latitude": 51.448765
      },
      {
        "longitude": -105.1162333,
        "latitude": 39.628574
      }
    ],
    "layer_ids": [
      "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0",
      "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0",
      "GLOBAL-1ARCSEC-NW_OFFSET-1in100-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0"
    ]
  }'

Expected response:

{
  "results": {
    "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0": {
      "values": [
        {
          "query_point": {
            "longitude": -2.60134,
            "latitude": 51.448765
          },
          "val": -32768
        },
        {
          "query_point": {
            "longitude": -105.1162333,
            "latitude": 39.628574
          },
          "val": 0
        }
      ]
    },
    "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0": {
      ...
    }
  }
}

Polygon queries

  • Polygons may not contain holes.
  • Polygons can either be simple_polygon objects or shapefiles.
  • The polygon's points must be specified in counterclockwise order.
  • The first and last points must be the same.
  • Consecutive duplicate points are not supported.
  • Polygons that cross the antimeridian are not supported. To retrieve data for this area, create separate polygons on either side of the antimeridian

To obtain data within an area defined by a polygon object:

layer_ids = [
    "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0",
    "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0"
]

poly = polygon(
    [
        point(51.45, 0),
        point(51.55, 0),
        point(51.55, -0.1),
        point(51.45, -0.1),
        point(51.45, 0),
    ]
)

results = client.geo.get_polygon(poly, layer_ids).results

# we return one tif per layer, this example saves the tif for each layer using the layer ID as the filename
for layer_id, data in results.items():
    f = open(layer_id+".tif", "wb")
    f.write(data.geo_tiff)
    f.close()

Request:

curl --location --request POST 'https://api.fathom.global/v2/polygon:data' \
  --header 'Content-Type: application/json' \
  --header 'Authorization: Bearer ACCESS_TOKEN' \
  --data-raw '{
    "layer_ids": [
      "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-PLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v3.0"
    ],
    "polygon": {
      "points": [
        {
          "latitude": 51.45,
          "longitude": 0
        },
        {
          "latitude": 51.55,
          "longitude": 0
        },
        {
          "latitude": 51.55,
          "longitude": -0.1
        },
        {
          "latitude": 51.45,
          "longitude": -0.1
        },
        {
          "latitude": 51.45,
          "longitude": 0
        }
      ]
    }
  }'

Response:

{
  "results": {
    "GLOBAL-1ARCSEC-NW_OFFSET-1in100-PLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v3.0": {
      "geo_tiff": "SUkqAAgAAAASAA..."
    }
  }
}

The geo_tiff element is a raw GeoTIFF file encoded as a base64-URL-encoded string. To write this to a file, first decode the base64 string then save the result to a file with a .tiff or .tif extension. An example for doing this in python is as follows:

encoded_string = response['results']['GLOBAL-1ARCSEC-NW_OFFSET-1in100-PLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v3.0']['geo_tiff']
encoded_bytes = encoded_string.encode('ascii')

decoded = base64.decodebytes(encoded_bytes)

with open(output_file_path, 'wb') as output_file:
    output_file.write(decoded)

Polygon stats queries

Allows fetching statistical data about a polygonal area. We currently return:

  • Mean
  • Max
  • Min
  • Standard Deviation
  • Share Above Zero

Polygons submitted to the polygon stats endpoint must follow the same rules as polygon queries.

poly = polygon(
    [
        point(52.41, -1.52),
        point(52.4, -1.52),
        point(52.4, -1.5),
        point(52.41, -1.5),
        point(52.41, -1.52),
    ]
)

layer_ids = [
    "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0",
    "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0",
    "GLOBAL-1ARCSEC-NW_OFFSET-1in100-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0"
]

data = client.geo.get_polygon_stats(poly, layer_ids)

Request:

curl --location --request POST 'https://api.fathom.global/v2/polygon:stats' \
  --header 'Content-Type: application/json' \
  --header 'Authorization: Bearer ACCESS_TOKEN' \
  --data-raw '{
    "polygon": {
      "points": [
        {
          "latitude": 51.45,
          "longitude": 0
        },
        {
          "latitude": 51.55,
          "longitude": 0
        },
        {
          "latitude": 51.55,
          "longitude": -0.1
        },
        {
          "latitude": 51.45,
          "longitude": -0.1
        },
        {
          "latitude": 51.45,
          "longitude": 0
        }
      ]
    },
    "layer_ids": [
      "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0",
      "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0",
      "GLOBAL-1ARCSEC-NW_OFFSET-1in100-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0"
    ]
  }'

Expected response:

{
  "results": {
    "US-1_3ARCSEC-00_OFFSET-1in1000-FLUVIAL-DEFENDED-DEPTH-2020-PERCENTILE50-v2.0.0": {
      "mean": 5,
      "max": 10,
      "min": 1,
      "stdDev": 2,
      "shareAboveZero": 0.6,
    },
    "GLOBAL-1ARCSEC-NW_OFFSET-1in1000-FLUVIAL-UNDEFENDED-DEPTH-2020-PERCENTILE50-v3.0": {
      ...
    }
  }
}

GeoJSON queries

The GeoJSON endpoints as documented in the REST reference and the Python SDK reference only accepts a subset of GeoJSON. In short:

  • Only 'Point', 'MultiPoint', and 'Polygon' are accepted
  • 'LineString', 'MultiPolygon', and 'MultiLineString' are not accepted
  • Inputs may be contained in a 'FeatureCollection' or 'Feature', but not 'GeometryCollection'
  • 'Polygon' geometries must follow the same rules as the rules for other endpoints

A non-exhaustive set of examples of valid inputs:

{
  "type": "Point",
  "coordinates": [-1.1081944, 50.8162643]
}
{
  "type": "MultiPoint",
  "coordinates": [
    [-1.1081944, 50.8162643],
    [-1.1071944, 50.8163643]
  ]
}
{
  "type": "Feature",
  "geometry": {
    "type": "Point",
    "coordinates": [-1.1081944, 50.8162643]
  },
  "properties": {
    "name": "Dinagat Islands"
  }
}
{
  "type": "Feature",
  "geometry": {
    "type": "Polygon",
    "coordinates": [
      [
        [100.0, 0.0],
        [101.0, 0.0],
        [101.0, 1.0],
        [100.0, 1.0],
        [100.0, 0.0]
      ]
    ]
  }
}
{
  "type": "FeatureCollection",
  "features": [
    {
      "type": "Feature",
      "geometry": {
        "type": "Point",
        "coordinates": [-1.1081944, 50.8162643]
      },
      "properties": {
        "name": "Point 1"
      }
    },
    {
      "type": "Feature",
      "geometry": {
        "type": "Point",
        "coordinates": [-1.1071944, 50.8163643]
      },
      "properties": {
        "name": "Point 2"
      }
    }
  ]
}

Note

If a FeatureCollection of Point or MultiPoint (or a combination) is given, the results from the API will be flattened into one response object. For the above 'FeatureCollection of Point' example, this will result in:

{
  "results": {
    "FLOOD_MAP-...": {
      "values": [
        {
          "query_point": {
            "longitude": -1.1081944,
            "latitude": 50.8162643
          },
          "val": 12
        },
        {
          "query_point": {
            "longitude": -1.1071944,
            "latitude": 50.8163643
          },
          "val": 56
        }
      ]
    }
  }
}

Some examples of invalid inputs:

Failure

LineString geometries are not allowed.

{
  "type": "Feature",
  "geometry": {
    "type": "LineString",
    "coordinates": [
      [102.0, 0.0],
      [103.0, 1.0],
      [104.0, 0.0],
      [105.0, 1.0]
    ]
  }
}

Failure

Polygons with multiple rings are not allowed.

{
  "type": "Feature",
  "geometry": {
    "type": "Polygon",
    "coordinates": [
      [
        [100.0, 0.0],
        [101.0, 0.0],
        [101.0, 1.0],
        [100.0, 1.0],
        [100.0, 0.0]
      ],
      [
        [100.0, 0.0],
        [101.0, 0.0],
        [101.0, 1.0],
        [100.0, 1.0],
        [100.0, 0.0]
      ]
    ]
  }
}

Failure

Only one polygon per request is allowed.

{
  "type": "FeatureCollection",
  "features": [
    {
      "type": "Feature",
      "geometry": {
        "type": "Polygon",
        "coordinates": [
          [
            [100.0, 0.0],
            [101.0, 0.0],
            [101.0, 1.0],
            [100.0, 1.0],
            [100.0, 0.0]
          ]
        ]
      }
    },
    {
      "type": "Feature",
      "geometry": {
        "type": "Polygon",
        "coordinates": [
          [
            [100.0, 0.0],
            [101.0, 0.0],
            [101.0, 1.0],
            [100.0, 1.0],
            [100.0, 0.0]
          ]
        ]
      }
    }
  ]
}

Vector file queries

The Python SDK supports reading polygons from a variety of vector files (see the SDK documentation for more information.)