Skip to content

Commit

Permalink
Handle custom granularities in source node to dataset conversion
Browse files Browse the repository at this point in the history
This will enable us to query the custom granularity columns from time spine tables.
  • Loading branch information
courtneyholcomb committed Sep 19, 2024
1 parent 4f15c43 commit 3fa44f8
Showing 1 changed file with 45 additions and 25 deletions.
70 changes: 45 additions & 25 deletions metricflow/dataset/convert_semantic_model.py
Original file line number Diff line number Diff line change
Expand Up @@ -105,15 +105,15 @@ def _create_time_dimension_instance(
self,
element_name: str,
entity_links: Tuple[EntityReference, ...],
time_granularity: TimeGranularity,
time_granularity: ExpandedTimeGranularity,
date_part: Optional[DatePart] = None,
semantic_model_name: Optional[str] = None,
) -> TimeDimensionInstance:
"""Create a time dimension instance from the dimension object from a semantic model in the model."""
time_dimension_spec = TimeDimensionSpec(
element_name=element_name,
entity_links=entity_links,
time_granularity=ExpandedTimeGranularity.from_time_granularity(time_granularity),
time_granularity=time_granularity,
date_part=date_part,
)

Expand Down Expand Up @@ -289,7 +289,7 @@ def _convert_time_dimension(
semantic_model_name=semantic_model_name,
element_name=dimension.reference.element_name,
entity_links=entity_links,
time_granularity=defined_time_granularity,
time_granularity=ExpandedTimeGranularity.from_time_granularity(defined_time_granularity),
)
time_dimension_instances.append(time_dimension_instance)

Expand All @@ -305,7 +305,7 @@ def _convert_time_dimension(
)
else:
select_columns.append(
self._build_column_for_time_granularity(
self._build_column_for_standard_time_granularity(
time_granularity=defined_time_granularity,
expr=dimension_select_expr,
column_alias=time_dimension_instance.associated_column.column_name,
Expand Down Expand Up @@ -340,12 +340,12 @@ def _build_time_dimension_instances_and_columns(
semantic_model_name=semantic_model_name,
element_name=element_name,
entity_links=entity_links,
time_granularity=time_granularity,
time_granularity=ExpandedTimeGranularity.from_time_granularity(time_granularity),
)
time_dimension_instances.append(time_dimension_instance)

select_columns.append(
self._build_column_for_time_granularity(
self._build_column_for_standard_time_granularity(
time_granularity=time_granularity,
expr=dimension_select_expr,
column_alias=time_dimension_instance.associated_column.column_name,
Expand All @@ -359,7 +359,7 @@ def _build_time_dimension_instances_and_columns(
semantic_model_name=semantic_model_name,
element_name=element_name,
entity_links=entity_links,
time_granularity=defined_time_granularity,
time_granularity=ExpandedTimeGranularity.from_time_granularity(defined_time_granularity),
date_part=date_part,
)
time_dimension_instances.append(time_dimension_instance)
Expand All @@ -373,7 +373,7 @@ def _build_time_dimension_instances_and_columns(

return (time_dimension_instances, select_columns)

def _build_column_for_time_granularity(
def _build_column_for_standard_time_granularity(
self, time_granularity: TimeGranularity, expr: SqlExpressionNode, column_alias: str
) -> SqlSelectColumn:
return SqlSelectColumn(
Expand Down Expand Up @@ -515,35 +515,55 @@ def create_sql_source_data_set(self, semantic_model: SemanticModel) -> SemanticM
def build_time_spine_source_data_set(self, time_spine_source: TimeSpineSource) -> SqlDataSet:
"""Build data set for time spine."""
from_source_alias = SequentialIdGenerator.create_next_id(StaticIdPrefix.TIME_SPINE_SOURCE).str_value
defined_time_granularity = time_spine_source.base_granularity
base_granularity = time_spine_source.base_granularity
time_column_name = time_spine_source.base_column

time_dimension_instances: List[TimeDimensionInstance] = []
select_columns: List[SqlSelectColumn] = []

time_dimension_instance = self._create_time_dimension_instance(
element_name=time_column_name, entity_links=(), time_granularity=defined_time_granularity
# Build base time dimension instances & columns
base_time_dimension_instance = self._create_time_dimension_instance(
element_name=time_column_name,
entity_links=(),
time_granularity=ExpandedTimeGranularity.from_time_granularity(base_granularity),
)
time_dimension_instances.append(time_dimension_instance)

dimension_select_expr = SemanticModelToDataSetConverter._make_element_sql_expr(
time_dimension_instances.append(base_time_dimension_instance)
base_dimension_select_expr = SemanticModelToDataSetConverter._make_element_sql_expr(
table_alias=from_source_alias, element_name=time_column_name
)
select_column = self._build_column_for_time_granularity(
time_granularity=defined_time_granularity,
expr=dimension_select_expr,
column_alias=time_dimension_instance.associated_column.column_name,
base_select_column = self._build_column_for_standard_time_granularity(
time_granularity=base_granularity,
expr=base_dimension_select_expr,
column_alias=base_time_dimension_instance.associated_column.column_name,
)
select_columns.append(select_column)

new_instances, new_columns = self._build_time_dimension_instances_and_columns(
defined_time_granularity=defined_time_granularity,
select_columns.append(base_select_column)
new_base_instances, new_base_columns = self._build_time_dimension_instances_and_columns(
defined_time_granularity=base_granularity,
element_name=time_column_name,
entity_links=(),
dimension_select_expr=dimension_select_expr,
dimension_select_expr=base_dimension_select_expr,
)
time_dimension_instances.extend(new_instances)
select_columns.extend(new_columns)
time_dimension_instances.extend(new_base_instances)
select_columns.extend(new_base_columns)

# Build custom granularity time dimension instances & columns
for custom_granularity in time_spine_source.custom_granularities:
custom_time_dimension_instance = self._create_time_dimension_instance(
element_name=time_column_name,
entity_links=(),
time_granularity=ExpandedTimeGranularity(
name=custom_granularity.name, base_granularity=base_granularity
),
)
time_dimension_instances.append(custom_time_dimension_instance)
custom_select_column = SqlSelectColumn(
expr=SemanticModelToDataSetConverter._make_element_sql_expr(
table_alias=from_source_alias,
element_name=custom_granularity.column_name or custom_granularity.name,
),
column_alias=custom_time_dimension_instance.associated_column.column_name,
)
select_columns.append(custom_select_column)

return SqlDataSet(
instance_set=InstanceSet(time_dimension_instances=tuple(time_dimension_instances)),
Expand Down

0 comments on commit 3fa44f8

Please sign in to comment.