Skip to main content

iNaturalist API client for python

Project description

pyinaturalist

Build Codecov Documentation

PyPI Conda PyPI - Python Versions

Run with Binder


Introduction

iNaturalist is a community science platform that helps people get involved in the natural world by observing and identifying the living things around them. Collectively, the community produces a rich source of global biodiversity data that can be valuable to anyone from hobbyists to scientists.

pyinaturalist is a client for the iNaturalist API that makes these data easily accessible in the python programming language.

Features

  • ➡️ Easier requests: Simplified request formats, easy pagination, and complete request parameter type annotations for better IDE integration
  • ⬅️ Convenient responses: Type conversions to the things you would expect in python, and typed model objects (Observation, Taxon, etc.) with full IDE autocompletion
  • 🔒 Security: Keyring integration for secure credential storage
  • 📗 Docs: Example requests, responses, scripts, and Jupyter notebooks to help get you started
  • 🧪 Testing: A dry-run testing mode to preview your requests before potentially modifying data
  • 💚 Responsible use: Follows the API Recommended Practices without extra configuration; caching and rate-limiting features reduce bandwidth usage, errors, and unexpected throttling

Supported Endpoints

Many of the most relevant API endpoints are supported, including:

  • 📝 Annotations and observation fields
  • 🆔 Identifications
  • 💬 Messages
  • 👀 Observations (multiple formats)
  • 📷 Observation photos + sounds
  • 📊 Observation histograms, observers, identifiers, life lists, and species counts
  • 📍 Places
  • 👥 Projects
  • 🐦 Species
  • 👤 Users

Quickstart

Here are usage examples for some of the most commonly used features.

First, install with pip:

pip install pyinaturalist

Then, import and create a client object. This will be our main interface to the API:

from pyinaturalist import *
client = iNatClient()

Note: If you are looking for the lower-level API functions (without the client class), see this page

Search observations

Let's start by searching for all your own observations. There are numerous fields you can search on, but we'll just use user_id for now:

>>> results = client.observations.search(user_id='my_username')

The full response consists of Observation objects with numerous attributes, but we can use pyinaturalist.pprint() to print out a condensed summary:

>>> pprint(results)
ID         Taxon                               Observed on   User     Location
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
117585709  Genus: Hyoscyamus (henbanes)        May 18, 2022  niconoe  Calvi, France
117464920  Genus: Omophlus                     May 17, 2022  niconoe  Galéria, France
117464393  Genus: Briza (Rattlesnake Grasses)  May 17, 2022  niconoe  Galéria, France
...

You can also get observation counts by species. On iNaturalist.org, this information can be found on the 'Species' tab of search results. For example, to get species counts of all your own research-grade observations:

>>> counts = client.observations.species_counts(user_id='my_username', quality_grade='research')
>>> pprint(counts)
 ID     Rank      Scientific name               Common name             Count
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
47934   species   🐛 Libellula luctuosa         Widow Skimmer           7
48627   species   🌻 Echinacea purpurea         Purple Coneflower       6
504060  species   🍄 Pleurotus citrinopileatus  Golden Oyster Mushroom  6
...

The data will be in the form of TaxonCount objects:

>>> counts[0]
TaxonCount(
    id=48662,
    name='Danaus plexippus',
    preferred_common_name='Monarch',
    rank='species',
    count=13,
    observations_count=458712,
    ...
)

Another useful format is the observation histogram, which shows the number of observations over a given time interval. The default is month_of_year:

>>> histogram = client.observations.histogram(user_id='my_username')
>>> pprint(histogram)
Month   Count
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
Jan     8       ████
Feb     1       
Mar     20      ██████████

The raw data will be a dict, with either int or datetime keys, depending on the interval:

>>> print(histogram.raw)
{
    1: 8,  # January
    2: 1,  # February
    3: 19, # March
    ...,   # etc.
}

Create and update observations: authentication

To create or modify observations, you will first need to log in. This requires creating an iNaturalist app, which will be used to get an access token.

creds = {
    'username': 'my_username',
    'password': 'my_password',
    'app_id': 'my_app_id',
    'app_secret': 'my_app_secret',
}
client = iNatClient(creds=creds)

See Authentication for more options including environment variables, keyrings, and password managers. A keyring is recommended, which does not require passing credentials directly:

client = iNatClient()
# Creds will be requested from the keyring when an authenticated request is made
client.observations.create(...)

Create and update observations

Now we can create a new observation:

from datetime import datetime

new_obs = client.observations.create(
    taxon_id=54327,  # Vespa Crabro
    observed_on_string=datetime.now(),
    time_zone='Brussels',
    description='This is a free text comment for the observation',
    tag_list='wasp, Belgium',
    latitude=50.647143,
    longitude=4.360216,
    positional_accuracy=50,  # GPS accuracy in meters
    photos=['~/observations/wasp1.jpg', '~/observations/wasp2.jpg'],
    sounds=['~/observations/recording.wav'],
)

We can then update the observation information, photos, or sounds:

client.observations.update(
    new_obs.id,  # Use the observation ID from the result above
    access_token=token,
    description='updated description !',
    photos='~/observations/wasp_nest.jpg',
    sounds='~/observations/wasp_nest.mp3',
)

Search species

There are many more resource types available besides observations. Taxonomy is another useful one.

Let's say you partially remember either a genus or family name that started with 'vespi'-something. The taxon search can be used to search by name, rank, and several other criteria:

>>> results = client.taxa.search(q='vespi', rank=['genus', 'family'])

As with observations, there is a lot of information available in the response (Taxon objects), but we'll print just a few basic details:

>>> pprint(results)
ID        Rank     Scientific name    Common name
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
52747     family   🐝 Vespidae        Hornets, Paper Wasps, Potter Wasps, and Allies
84737     genus    🦋 Vespina
646195    genus    🪰 Vespiodes
...

Next Steps

For more information, see:

  • User Guide: introduction and general features that apply to most endpoints
  • Endpoint Summary: a complete list of endpoints wrapped by pyinaturalist
  • Examples: data visualizations and other examples of things to do with iNaturalist data
  • Reference: Detailed API documentation
  • Contributing Guide: development details for anyone interested in contributing to pyinaturalist
  • History: details on past and current releases
  • Issues: planned & proposed features

Feedback

If you have any problems, suggestions, or questions about pyinaturalist, you are welcome to create an issue or discussion. Also, PRs are welcome!

Note: pyinaturalist is developed by members of the iNaturalist community, and is not endorsed by iNaturalist.org or the California Academy of Sciences. If you have non-python-specific questions about the iNaturalist API or iNaturalist in general, the iNaturalist Community Forum is the best place to start.

Related Projects

Other python projects related to iNaturalist:

  • naturtag: A desktop application for tagging image files with iNaturalist taxonomy & observation metadata
  • pyinaturalist-convert: Tools to convert observation data to and from a variety of useful formats
  • pyinaturalist-notebook: Jupyter notebook Docker image for pyinaturalist
  • dronefly: A Discord bot with iNaturalist integration, used by the iNaturalist Discord server.

Project details


Release history Release notifications | RSS feed

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyinaturalist-1.0.0.dev1.tar.gz (112.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

pyinaturalist-1.0.0.dev1-py3-none-any.whl (154.5 kB view details)

Uploaded Python 3

File details

Details for the file pyinaturalist-1.0.0.dev1.tar.gz.

File metadata

  • Download URL: pyinaturalist-1.0.0.dev1.tar.gz
  • Upload date:
  • Size: 112.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for pyinaturalist-1.0.0.dev1.tar.gz
Algorithm Hash digest
SHA256 b8bfd398d0c9a5e27b8cd36cb8f4520432a16e17573a5f7b72cc799a759a81c4
MD5 32a84e4bacca293965f44d39f5decd7d
BLAKE2b-256 3a9ca268b3df971dca65f60850400aad0dd1a0074541faec1912ec9f8f4e21f9

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyinaturalist-1.0.0.dev1.tar.gz:

Publisher: deploy.yml on pyinat/pyinaturalist

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pyinaturalist-1.0.0.dev1-py3-none-any.whl.

File metadata

File hashes

Hashes for pyinaturalist-1.0.0.dev1-py3-none-any.whl
Algorithm Hash digest
SHA256 eb61146b5ac82c346b2881defbda602850671b8e65410a708c0cfdde72c15190
MD5 58a922c6a52fcd5989cfdc0e2635d50a
BLAKE2b-256 19d2bd24dd9ab3563477c562a1b9e99a4d1ea63da4f72b39ee6836bbea7eb40e

See more details on using hashes here.

Provenance

The following attestation bundles were made for pyinaturalist-1.0.0.dev1-py3-none-any.whl:

Publisher: deploy.yml on pyinat/pyinaturalist

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page