-
Notifications
You must be signed in to change notification settings - Fork 216
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add proportional staging index dag (#3763)
* Add API connection * Pull out common elasticsearch functions * Remove duplicated tasks * Add new dag * Tests * Add tasks for point alias * Update dag docs * Fix tests * Fix es query for reindexing * Get stats for source index rather than using stats endpoint * Raise error when incorrect number of documents are reindexed * Prevent scroll from being disabled and causing error with auto-slicing * Turn off slicing for reindex with max_docs instead * Do not perform multiple source reindexes simultaneously * Update docs to correctly describe source index as a staging index - Since we're reindexing from a staging index, it's also okay to run the reindexing tasks in parallel - Update broken import * Lint * Log reindexed count * Clarify language to 'staging', simplify math * Add clarifying comment about aggregations query * Make minimum of 0 exclusive * Only refresh index once, clean up es task params * Clarify arguments to get_staging_source_counts
- Loading branch information
Showing
10 changed files
with
704 additions
and
163 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,225 @@ | ||
import logging | ||
from datetime import timedelta | ||
from typing import Literal, Union | ||
|
||
from airflow.decorators import task, task_group | ||
from airflow.models.connection import Connection | ||
from airflow.providers.elasticsearch.hooks.elasticsearch import ElasticsearchPythonHook | ||
from airflow.sensors.base import PokeReturnValue | ||
from airflow.utils.trigger_rule import TriggerRule | ||
|
||
from common.constants import REFRESH_POKE_INTERVAL | ||
|
||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
# Index settings that should not be copied over from the base configuration when | ||
# creating a new index. | ||
EXCLUDED_INDEX_SETTINGS = {"provided_name", "creation_date", "uuid", "version"} | ||
|
||
|
||
@task | ||
def get_es_host(environment: str): | ||
es_conn = Connection.get_connection_from_secrets( | ||
f"elasticsearch_http_{environment}" | ||
) | ||
return es_conn.get_uri() | ||
|
||
|
||
@task | ||
def get_index_configuration( | ||
source_index: str, | ||
es_host: str, | ||
): | ||
""" | ||
Return the configuration for the index identified by the | ||
`source_index` param. `source_index` may be either an index name | ||
or an alias, but must uniquely identify one existing index or an | ||
error will be raised. | ||
""" | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
|
||
response = es_conn.indices.get( | ||
index=source_index, | ||
# Return empty dict instead of throwing error if no index can be | ||
# found. We raise our own error instead. | ||
ignore_unavailable=True, | ||
) | ||
|
||
if len(response) != 1: | ||
raise ValueError(f"Index {source_index} could not be uniquely identified.") | ||
|
||
# The response has the form: | ||
# { index_name: index_configuration } | ||
# However, since `source_index` can be an alias rather than the index name, | ||
# we do not necessarily know the index_name so we cannot access the configuration | ||
# directly by key. We instead get the first value from the dict, knowing that we | ||
# have already ensured in a previous check that there is exactly one value in the | ||
# response. | ||
config = next(iter(response.values())) | ||
return config | ||
|
||
|
||
def remove_excluded_index_settings(index_config): | ||
""" | ||
Remove fields from the given index configuration that should not be included when | ||
using it to create a new index. | ||
""" | ||
# Remove fields from the current_index_config that should not be copied | ||
# over into the new index (such as uuid) | ||
for setting in EXCLUDED_INDEX_SETTINGS: | ||
index_config.get("settings", {}).get("index", {}).pop(setting) | ||
|
||
# Aliases should also not by applied automatically | ||
index_config.pop("aliases") | ||
|
||
return index_config | ||
|
||
|
||
@task | ||
def create_index(index_config, es_host: str): | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
|
||
new_index = es_conn.indices.create(**index_config) | ||
|
||
return new_index | ||
|
||
|
||
@task_group(group_id="trigger_and_wait_for_reindex") | ||
def trigger_and_wait_for_reindex( | ||
es_host: str, | ||
destination_index: str, | ||
source_index: str, | ||
timeout: timedelta, | ||
requests_per_second: int, | ||
query: dict | None = None, | ||
max_docs: int | None = None, | ||
refresh: bool = True, | ||
slices: Union[int, Literal["auto"]] = "auto", | ||
): | ||
@task | ||
def trigger_reindex( | ||
es_host: str, | ||
destination_index: str, | ||
source_index: str, | ||
query: dict, | ||
requests_per_second: int, | ||
max_docs: int | None, | ||
slices: Union[int, Literal["auto"]], | ||
): | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
source = {"index": source_index} | ||
# An empty query is not accepted; only pass it | ||
# if a query was actually supplied | ||
if query: | ||
source["query"] = query | ||
|
||
response = es_conn.reindex( | ||
source=source, | ||
dest={"index": destination_index}, | ||
max_docs=max_docs, | ||
# Parallelize indexing when not None | ||
slices=slices, | ||
# Do not hold the slot while awaiting completion | ||
wait_for_completion=False, | ||
# Whether to immediately refresh the index after completion to make | ||
# the data available for search | ||
refresh=refresh, | ||
# Throttle | ||
requests_per_second=requests_per_second, | ||
) | ||
return response["task"] | ||
|
||
@task.sensor( | ||
poke_interval=REFRESH_POKE_INTERVAL, timeout=timeout, mode="reschedule" | ||
) | ||
def wait_for_reindex( | ||
es_host: str, task_id: str, expected_docs: int | None = None | ||
) -> PokeReturnValue: | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
|
||
response = es_conn.tasks.get(task_id=task_id) | ||
|
||
count = response.get("task", {}).get("status", {}).get("total") | ||
if expected_docs and count != expected_docs: | ||
logger.info( | ||
f"Reindexed {count} documents, but {expected_docs} were expected." | ||
) | ||
else: | ||
logger.info(f"Reindexed {count} documents.") | ||
|
||
return PokeReturnValue(is_done=response.get("completed") is True) | ||
|
||
trigger_reindex_task = trigger_reindex( | ||
es_host, | ||
destination_index, | ||
source_index, | ||
query, | ||
requests_per_second, | ||
max_docs, | ||
slices, | ||
) | ||
|
||
wait_for_reindex_task = wait_for_reindex( | ||
task_id=trigger_reindex_task, expected_docs=max_docs, es_host=es_host | ||
) | ||
|
||
trigger_reindex_task >> wait_for_reindex_task | ||
|
||
|
||
@task | ||
def refresh_index(es_host: str, index_name: str): | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
return es_conn.indices.refresh(index=index_name) | ||
|
||
|
||
@task_group(group_id="point_alias") | ||
def point_alias(index_name: str, alias: str, es_host: str): | ||
""" | ||
Point the target alias to the given index. If the alias is already being | ||
used by one or more indices, it will first be removed from all of them. | ||
""" | ||
|
||
@task.branch | ||
def check_if_alias_exists(alias: str, es_host: str): | ||
"""Check if the alias already exists.""" | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
return ( | ||
"point_alias.remove_existing_alias" | ||
if es_conn.indices.exists_alias(name=alias) | ||
else "point_alias.point_new_alias" | ||
) | ||
|
||
@task | ||
def remove_existing_alias(alias: str, es_host: str): | ||
"""Remove the given alias from any indices to which it points.""" | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
response = es_conn.indices.delete_alias( | ||
name=alias, | ||
# Remove the alias from _all_ indices to which it currently | ||
# applies | ||
index="_all", | ||
) | ||
return response.get("acknowledged") | ||
|
||
@task | ||
def point_new_alias( | ||
es_host: str, | ||
index_name: str, | ||
alias: str, | ||
): | ||
es_conn = ElasticsearchPythonHook(hosts=[es_host]).get_conn | ||
response = es_conn.indices.put_alias(index=index_name, name=alias) | ||
return response.get("acknowledged") | ||
|
||
exists_alias = check_if_alias_exists(alias, es_host) | ||
remove_alias = remove_existing_alias(alias, es_host) | ||
|
||
point_alias = point_new_alias.override( | ||
# The remove_alias task may be skipped. | ||
trigger_rule=TriggerRule.NONE_FAILED, | ||
)(es_host, index_name, alias) | ||
|
||
exists_alias >> [remove_alias, point_alias] | ||
remove_alias >> point_alias |
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
Oops, something went wrong.