Skip to content

Commit

Permalink
Consolidate logic around choosing appropriate time spine source
Browse files Browse the repository at this point in the history
This will change later when we support multiple time spine nodes per query. For now, move the error to the core function so that we don't need to do error handling everywhere this gets used (which will be several places further up the stack). Also adds a helper to improve readability.
  • Loading branch information
courtneyholcomb committed Nov 23, 2024
1 parent 9faa23e commit 9633e66
Show file tree
Hide file tree
Showing 3 changed files with 19 additions and 19 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -99,10 +99,10 @@ def build_custom_granularities(time_spine_sources: Sequence[TimeSpineSource]) ->
}

@staticmethod
def choose_time_spine_sources(
def choose_time_spine_source(
required_time_spine_specs: Sequence[TimeDimensionSpec],
time_spine_sources: Dict[TimeGranularity, TimeSpineSource],
) -> Sequence[TimeSpineSource]:
) -> TimeSpineSource:
"""Determine which time spine sources to use to satisfy the given specs.
Custom grains can only use the time spine where they are defined. For standard grains, this will choose the time
Expand Down Expand Up @@ -145,7 +145,13 @@ def choose_time_spine_sources(
if not required_time_spines.intersection(set(compatible_time_spines_for_standard_grains.values())):
required_time_spines.add(time_spine_sources[max(compatible_time_spines_for_standard_grains)])

return tuple(required_time_spines)
if len(required_time_spines) != 1:
raise RuntimeError(
"Multiple time spines are required to satisfy the specs, but only one is supported per query currently. "
f"Multiple will be supported in the future. Time spines required: {required_time_spines}."
)

return required_time_spines.pop()

@property
def data_set_description(self) -> str:
Expand Down
18 changes: 9 additions & 9 deletions metricflow/dataflow/builder/dataflow_plan_builder.py
Original file line number Diff line number Diff line change
Expand Up @@ -1037,15 +1037,8 @@ def _find_source_node_recipe_non_cached(
)
# If metric_time is requested without metrics, choose appropriate time spine node to select those values from.
if linkable_specs_to_satisfy.metric_time_specs:
time_spine_sources = TimeSpineSource.choose_time_spine_sources(
required_time_spine_specs=linkable_specs_to_satisfy.metric_time_specs,
time_spine_sources=self._source_node_builder.time_spine_sources,
)
assert len(time_spine_sources) == 1, (
"Exactly one time spine source should have been selected for base grains."
"This indicates internal misconfiguration."
)
time_spine_node = self._source_node_set.time_spine_nodes[time_spine_sources[0].base_granularity]
time_spine_source = self._choose_time_spine_source(linkable_specs_to_satisfy.metric_time_specs)
time_spine_node = self._source_node_set.time_spine_nodes[time_spine_source.base_granularity]
candidate_nodes_for_right_side_of_join += [time_spine_node]
candidate_nodes_for_left_side_of_join += [time_spine_node]
default_join_type = SqlJoinType.FULL_OUTER
Expand Down Expand Up @@ -1812,3 +1805,10 @@ def _build_semi_additive_join_node(
agg_by_function=non_additive_dimension_spec.window_choice,
queried_time_dimension_spec=queried_time_dimension_spec,
)

def _choose_time_spine_source(self, required_time_spine_specs: Sequence[TimeDimensionSpec]) -> TimeSpineSource:
"""Choose the time spine source that can satisfy the required time spine specs."""
return TimeSpineSource.choose_time_spine_source(
required_time_spine_specs=required_time_spine_specs,
time_spine_sources=self._source_node_builder.time_spine_sources,
)
8 changes: 1 addition & 7 deletions metricflow/plan_conversion/dataflow_to_sql.py
Original file line number Diff line number Diff line change
Expand Up @@ -333,15 +333,9 @@ def _make_time_spine_data_set(
]
required_specs = queried_specs + specs_required_for_where_constraints

time_spine_sources = TimeSpineSource.choose_time_spine_sources(
time_spine_source = TimeSpineSource.choose_time_spine_source(
required_time_spine_specs=required_specs, time_spine_sources=self._time_spine_sources
)
# TODO: handle multiple time spine joins
assert len(time_spine_sources) == 1, (
"Join to time spine with custom granularity currently only supports one custom granularity per query. "
"Full feature coming soon."
)
time_spine_source = time_spine_sources[0]
time_spine_base_granularity = ExpandedTimeGranularity.from_time_granularity(time_spine_source.base_granularity)

base_column_expr = SqlColumnReferenceExpression.from_table_and_column_names(
Expand Down

0 comments on commit 9633e66

Please sign in to comment.