This repository has been archived by the owner on Aug 4, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 54
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* templated changes from existing script/readme * first stab at transformation using pandas * add capability to load .gz files from s3 * load data from S3 and store transformation logic * clean up old file structure and files * add test source files * cleanup file structure * get sample data into minio container * add sql transformation steps * hide minio testing side effects from git * add python wrappers for airflow and testing * fyi on the source of the sample data * Minor tweaks to get this working locally * Update .gitignore Use exclude so only need to list the one provider subfolder Co-authored-by: Madison Swain-Bowden <[email protected]> * Linting * add separate s3 load step to dev docker setup * airflow loads s3 & calls data ingestor to save tsv * make bucket loading piece readable * remove unnecessary gitignore change * put back basic minio volume * fix count typo in testing comment * linting sql files from sql fluff * fixing linting capitalization error * strengthen file references Co-authored-by: Madison Swain-Bowden <[email protected]> * move pg hook to init to reduce overhead * Aggregate dependencies into reusable just recipe, add load_to_s3 * Add check file exists on S3 to DAG * fix ingester class initialization override * stop overwriting pg temp files * fix typo in order of callables * dag runs with python pagination, though tests fail * sql for python pagination * create and manage larger test data files * performant SQL to export JSON * landing url not langing url * temp file name with timestamp for postgres * inaturalist provider ingester + testing * removed likely to error command * committing workflow for review, not complete * use batch_limit * complete the workflow cycle * make preingestion factory logic more concise * check s3 file update date to run dag daily * testing with resource files instead of minio * more concise batch data testing for none * monthly run, checking if files have been updated * linting * pass tags as a list * list reformat linting * more readable bash command Co-authored-by: Madison Swain-Bowden <[email protected]> * inaturalist --> iNaturalist in class name * old_query_params --> prev_query_params * improved description for sample data file manager Co-authored-by: Madison Swain-Bowden <[email protected]> * parametrize tests for missing data Co-authored-by: Madison Swain-Bowden <[email protected]> * more concise / non-redundant get usage Co-authored-by: Olga Bulat <[email protected]> * streamline checking for new data * remove unnecessary logging and imports * iNaturalistDataIngester -> INaturalistDataIngester * don't need to make foreign ID a string up front * clarify comment on observer FK Co-authored-by: sarayourfriend <[email protected]> * more readable use of sql template * streamline transformations in postgres * add test case with no ancestry info / tags * linting test db response * remove env check because file won't be in prod * Add SQL tests * remove universal tags * adjusted db response without "life" tag * lint db response * remove life from hard coded test results * drop source tables and schema after ingestion * nicer feedback on initial load test with pytest * try adding default image category * Print logs on failure * Revert "Print logs on failure" This reverts commit 9eed1f9. * post-ingestion task comment * clarify and simplify test * simplify sql names and remove dedent * clarify duplicate photo id status * lint fix * documentation for pre- and post-ingestion task params * Update DAGs.md * add issue references to comments * fix comment typo Co-authored-by: Olga Bulat <[email protected]> Co-authored-by: Madison Swain-Bowden <[email protected]> Co-authored-by: Olga Bulat <[email protected]> Co-authored-by: sarayourfriend <[email protected]>
- Loading branch information
1 parent
2d41485
commit ee474f2
Showing
22 changed files
with
854 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
166 changes: 166 additions & 0 deletions
166
openverse_catalog/dags/providers/provider_api_scripts/inaturalist.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,166 @@ | ||
""" | ||
Provider: iNaturalist | ||
Output: TSV file containing the media metadata. | ||
Notes: [The iNaturalist API is not intended for data scraping.] | ||
(https://api.inaturalist.org/v1/docs/) | ||
[But there is a full dump intended for sharing on S3.] | ||
(https://github.com/inaturalist/inaturalist-open-data/tree/documentation/Metadata) | ||
Because these are very large normalized tables, as opposed to more document | ||
oriented API responses, we found that bringing the data into postgres first | ||
was the most effective approach. [More detail in slack here.] | ||
(https://wordpress.slack.com/archives/C02012JB00N/p1653145643080479?thread_ts=1653082292.714469&cid=C02012JB00N) | ||
We use the table structure defined [here,] | ||
(https://github.com/inaturalist/inaturalist-open-data/blob/main/Metadata/structure.sql) | ||
except for adding ancestry tags to the taxa table. | ||
""" | ||
|
||
import os | ||
from pathlib import Path | ||
from typing import Dict | ||
|
||
import pendulum | ||
from airflow.exceptions import AirflowSkipException | ||
from airflow.operators.python import PythonOperator | ||
from airflow.providers.amazon.aws.hooks.s3 import S3Hook | ||
from airflow.providers.postgres.hooks.postgres import PostgresHook | ||
from airflow.providers.postgres.operators.postgres import PostgresOperator | ||
from airflow.utils.task_group import TaskGroup | ||
from common.constants import POSTGRES_CONN_ID | ||
from common.licenses import LicenseInfo, get_license_info | ||
from common.loader import provider_details as prov | ||
from providers.provider_api_scripts.provider_data_ingester import ProviderDataIngester | ||
|
||
|
||
AWS_CONN_ID = os.getenv("AWS_CONN_ID", "test_conn_id") | ||
PROVIDER = prov.INATURALIST_DEFAULT_PROVIDER | ||
SCRIPT_DIR = Path(__file__).parents[1] / "provider_csv_load_scripts/inaturalist" | ||
SOURCE_FILE_NAMES = ["photos", "observations", "taxa", "observers"] | ||
|
||
|
||
class INaturalistDataIngester(ProviderDataIngester): | ||
|
||
providers = {"image": prov.INATURALIST_DEFAULT_PROVIDER} | ||
|
||
def __init__(self, *kwargs): | ||
super(INaturalistDataIngester, self).__init__() | ||
self.pg = PostgresHook(POSTGRES_CONN_ID) | ||
|
||
# adjustments to buffer limits. TO DO: try to integrate this with the dev | ||
# environment logic in the base class, rather than just over-writing it. | ||
# See https://github.com/WordPress/openverse-catalog/issues/682 | ||
self.media_stores["image"].buffer_length = 10_000 | ||
self.batch_limit = 10_000 | ||
self.sql_template = (SCRIPT_DIR / "export_to_json.template.sql").read_text() | ||
|
||
def get_next_query_params(self, prev_query_params=None, **kwargs): | ||
if prev_query_params is None: | ||
return {"offset_num": 0} | ||
else: | ||
next_offset = prev_query_params["offset_num"] + self.batch_limit | ||
return {"offset_num": next_offset} | ||
|
||
def get_response_json(self, query_params: Dict): | ||
""" | ||
Call the SQL to pull json from Postgres, where the raw data has been loaded. | ||
""" | ||
sql_string = self.sql_template.format( | ||
batch_limit=self.batch_limit, offset_num=query_params["offset_num"] | ||
) | ||
sql_result = self.pg.get_records(sql_string) | ||
# Postgres returns a list of tuples, even if it's one tuple with one item. | ||
return sql_result[0][0] | ||
|
||
def get_batch_data(self, response_json): | ||
if response_json: | ||
return response_json | ||
return None | ||
|
||
def get_record_data(self, data): | ||
if data.get("foreign_identifier") is None: | ||
# TO DO: maybe raise an error here or after a certain number of these? | ||
# more info in https://github.com/WordPress/openverse-catalog/issues/684 | ||
return None | ||
license_url = data.get("license_url") | ||
license_info = get_license_info(license_url=license_url) | ||
if license_info == LicenseInfo(None, None, None, None): | ||
return None | ||
record_data = {k: data[k] for k in data.keys() if k != "license_url"} | ||
record_data["license_info"] = license_info | ||
return record_data | ||
|
||
def get_media_type(self, record): | ||
# This provider only supports Images via S3, though they have some audio files | ||
# on site and in the API. | ||
return "image" | ||
|
||
def endpoint(self): | ||
raise NotImplementedError("Normalized TSV files from AWS S3 means no endpoint.") | ||
|
||
@staticmethod | ||
def compare_update_dates( | ||
last_success: pendulum.DateTime | None, s3_keys: list, aws_conn_id=AWS_CONN_ID | ||
): | ||
# if it was never run, assume the data is new | ||
if last_success is None: | ||
return | ||
s3 = S3Hook(aws_conn_id=aws_conn_id) | ||
s3_client = s3.get_client_type() | ||
for key in s3_keys: | ||
# this will error out if the files don't exist, and bubble up as an | ||
# informative failure | ||
last_modified = s3_client.head_object( | ||
Bucket="inaturalist-open-data", Key=key | ||
)["LastModified"] | ||
# if any file has been updated, let's pull them all | ||
if last_success < last_modified: | ||
return | ||
# If no files have been updated, skip the DAG | ||
raise AirflowSkipException("Nothing new to ingest") | ||
|
||
@staticmethod | ||
def create_preingestion_tasks(): | ||
|
||
with TaskGroup(group_id="preingestion_tasks") as preingestion_tasks: | ||
|
||
check_for_file_updates = PythonOperator( | ||
task_id="check_for_file_updates", | ||
python_callable=INaturalistDataIngester.compare_update_dates, | ||
op_kwargs={ | ||
"last_success": "{{ prev_start_date_success }}", | ||
"s3_keys": [ | ||
f"{file_name}.csv.gz" for file_name in SOURCE_FILE_NAMES | ||
], | ||
}, | ||
) | ||
|
||
create_inaturalist_schema = PostgresOperator( | ||
task_id="create_inaturalist_schema", | ||
postgres_conn_id=POSTGRES_CONN_ID, | ||
sql=(SCRIPT_DIR / "create_schema.sql").read_text(), | ||
) | ||
|
||
with TaskGroup(group_id="load_source_files") as load_source_files: | ||
for source_name in SOURCE_FILE_NAMES: | ||
PostgresOperator( | ||
task_id=f"load_{source_name}", | ||
postgres_conn_id=POSTGRES_CONN_ID, | ||
sql=(SCRIPT_DIR / f"{source_name}.sql").read_text(), | ||
), | ||
|
||
(check_for_file_updates >> create_inaturalist_schema >> load_source_files) | ||
|
||
return preingestion_tasks | ||
|
||
@staticmethod | ||
def create_postingestion_tasks(): | ||
drop_inaturalist_schema = PostgresOperator( | ||
task_id="drop_inaturalist_schema", | ||
postgres_conn_id=POSTGRES_CONN_ID, | ||
sql="DROP SCHEMA IF EXISTS inaturalist CASCADE", | ||
) | ||
return drop_inaturalist_schema |
4 changes: 4 additions & 0 deletions
4
openverse_catalog/dags/providers/provider_csv_load_scripts/inaturalist/create_schema.sql
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
CREATE SCHEMA IF NOT EXISTS inaturalist; | ||
COMMIT; | ||
SELECT schema_name | ||
FROM information_schema.schemata WHERE schema_name = 'inaturalist'; |
Oops, something went wrong.