Skip to content

Commit

Permalink
Remove unused only_apply_for_metric_time param from MinimumTimeGrainP…
Browse files Browse the repository at this point in the history
…attern
  • Loading branch information
courtneyholcomb committed Jul 12, 2024
1 parent a2c778d commit bd00e02
Show file tree
Hide file tree
Showing 2 changed files with 70 additions and 23 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
from __future__ import annotations

from collections import defaultdict
from typing import Dict, List, Sequence, Set

from dbt_semantic_interfaces.type_enums import TimeGranularity
from typing_extensions import override

from metricflow_semantics.specs.patterns.spec_pattern import SpecPattern
from metricflow_semantics.specs.spec_classes import (
InstanceSpec,
LinkableInstanceSpec,
TimeDimensionSpec,
TimeDimensionSpecComparisonKey,
TimeDimensionSpecField,
)
from metricflow_semantics.specs.spec_set import group_specs_by_type


class MinimumTimeGrainPattern(SpecPattern):
"""A pattern that matches linkable specs, but for time dimension specs, only the one with the finest grain.
e.g.
inputs:
[
TimeDimensionSpec('metric_time', 'day'),
TimeDimensionSpec('metric_time', 'month.'),
DimensionSpec('listing__country'),
]
matches:
[
TimeDimensionSpec('metric_time', 'day'),
DimensionSpec('listing__country'),
]
The finest grain represents the defined grain of the time dimension in the semantic model when evaluating specs
of the source.
This pattern helps to implement matching of group-by-items for where filters - in those cases, an ambiguously
specified group-by-item can only match to time dimension spec with the base grain.
Also, this is currently used to help implement restrictions on cumulative metrics where they can only be queried
by the base grain of metric_time.
"""

@override
def match(self, candidate_specs: Sequence[InstanceSpec]) -> Sequence[InstanceSpec]:
spec_set = group_specs_by_type(candidate_specs)

spec_key_to_grains: Dict[TimeDimensionSpecComparisonKey, Set[TimeGranularity]] = defaultdict(set)
spec_key_to_specs: Dict[TimeDimensionSpecComparisonKey, List[TimeDimensionSpec]] = defaultdict(list)
for time_dimension_spec in spec_set.time_dimension_specs:
spec_key = time_dimension_spec.comparison_key(exclude_fields=(TimeDimensionSpecField.TIME_GRANULARITY,))
spec_key_to_grains[spec_key].add(time_dimension_spec.time_granularity)
spec_key_to_specs[spec_key].append(time_dimension_spec)

matched_time_dimension_specs: List[TimeDimensionSpec] = []
for spec_key, time_grains in spec_key_to_grains.items():
matched_time_dimension_specs.append(spec_key_to_specs[spec_key][0].with_grain(min(time_grains)))

matching_specs: Sequence[LinkableInstanceSpec] = (
spec_set.dimension_specs
+ tuple(matched_time_dimension_specs)
+ spec_set.entity_specs
+ spec_set.group_by_metric_specs
)

return matching_specs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
from dbt_semantic_interfaces.type_enums import TimeGranularity
from typing_extensions import override

from metricflow_semantics.specs.patterns.metric_time_pattern import MetricTimePattern
from metricflow_semantics.specs.patterns.spec_pattern import SpecPattern
from metricflow_semantics.specs.spec_classes import (
InstanceSpec,
Expand Down Expand Up @@ -41,32 +40,10 @@ class MinimumTimeGrainPattern(SpecPattern):
This pattern helps to implement matching of group-by-items for where filters - in those cases, an ambiguously
specified group-by-item can only match to time dimension spec with the base grain.
Also, this is currently used to help implement restrictions on cumulative metrics where they can only be queried
by the base grain of metric_time.
"""

def __init__(self, only_apply_for_metric_time: bool = False) -> None:
"""Initializer.
Args:
only_apply_for_metric_time: If set, only remove time dimension specs with a non-base grain if it's for
metric_time. This parameter is useful for implementing restrictions on cumulative metrics as they can only
be queried by the base grain of metric_time.
TODO: This is a little odd. This can be replaced once composite patterns are supported.
"""
self._only_apply_for_metric_time = only_apply_for_metric_time

@override
def match(self, candidate_specs: Sequence[InstanceSpec]) -> Sequence[InstanceSpec]:
if self._only_apply_for_metric_time:
metric_time_specs = MetricTimePattern().match(candidate_specs)
other_specs = tuple(spec for spec in candidate_specs if spec not in metric_time_specs)

return other_specs + tuple(
MinimumTimeGrainPattern(only_apply_for_metric_time=False).match(metric_time_specs)
)

spec_set = group_specs_by_type(candidate_specs)

spec_key_to_grains: Dict[TimeDimensionSpecComparisonKey, Set[TimeGranularity]] = defaultdict(set)
Expand Down

0 comments on commit bd00e02

Please sign in to comment.