From 07d9d5d516aa86f981d068578969d8f74741e150 Mon Sep 17 00:00:00 2001 From: Courtney Holcomb Date: Tue, 7 Nov 2023 14:52:43 -0800 Subject: [PATCH 1/2] Script for populating all persistent source schemas --- Makefile | 5 +++ metricflow/test/generate_snapshots.py | 18 +++++--- .../populate_persistent_source_schemas.py | 45 +++++++++++++++++++ 3 files changed, 63 insertions(+), 5 deletions(-) create mode 100644 metricflow/test/populate_persistent_source_schemas.py diff --git a/Makefile b/Makefile index e70cbb94ea..41eb3d1d16 100644 --- a/Makefile +++ b/Makefile @@ -74,6 +74,11 @@ postgresql postgres: regenerate-test-snapshots: hatch -v run dev-env:python metricflow/test/generate_snapshots.py +# Populate persistent source schemas for all relevant SQL engines. +.PHONY: populate-persistent-source-schemas +populate-persistent-source-schemas: + hatch -v run dev-env:python metricflow/test/populate_persistent_source_schemas.py + # Re-generate snapshots for the default SQL engine. .PHONY: test-snap test-snap: diff --git a/metricflow/test/generate_snapshots.py b/metricflow/test/generate_snapshots.py index ceb1568e84..3be91cb750 100644 --- a/metricflow/test/generate_snapshots.py +++ b/metricflow/test/generate_snapshots.py @@ -37,7 +37,7 @@ import logging import os from dataclasses import dataclass -from typing import Optional, Sequence +from typing import Callable, Optional, Sequence from dbt_semantic_interfaces.enum_extension import assert_values_exhausted from dbt_semantic_interfaces.implementations.base import FrozenBaseModel @@ -107,7 +107,11 @@ def run_command(command: str) -> None: # noqa: D raise RuntimeError(f"Error running command: {command}") -def run_tests(test_configuration: MetricFlowTestConfiguration) -> None: # noqa: D +def set_engine_env_variables(test_configuration: MetricFlowTestConfiguration) -> None: + """Set connection env variables dynamically for the engine being used. + + Requires MF_TEST_ENGINE_CREDENTIALS env variable to be set with creds for all engines. + """ if test_configuration.credential_set.engine_url is None: if "MF_SQL_ENGINE_URL" in os.environ: del os.environ["MF_SQL_ENGINE_URL"] @@ -120,6 +124,10 @@ def run_tests(test_configuration: MetricFlowTestConfiguration) -> None: # noqa: else: os.environ["MF_SQL_ENGINE_PASSWORD"] = test_configuration.credential_set.engine_password + +def run_tests(test_configuration: MetricFlowTestConfiguration) -> None: # noqa: D + set_engine_env_variables(test_configuration) + if test_configuration.engine is SqlEngine.DUCKDB: # DuckDB is fast, so generate all snapshots, including the engine-agnostic ones run_command(f"pytest -x -vv -n 4 --overwrite-snapshots -k 'not itest' {TEST_DIRECTORY}") @@ -145,7 +153,7 @@ def run_tests(test_configuration: MetricFlowTestConfiguration) -> None: # noqa: assert_values_exhausted(test_configuration.engine) -def run_cli() -> None: # noqa: D +def run_cli(function_to_run: Callable) -> None: # noqa: D # Setup logging. dev_format = "%(asctime)s %(levelname)s %(filename)s:%(lineno)d [%(threadName)s] - %(message)s" logging.basicConfig(level=logging.INFO, format=dev_format) @@ -165,8 +173,8 @@ def run_cli() -> None: # noqa: D logger.info( f"Running tests for {test_configuration.engine} with URL: {test_configuration.credential_set.engine_url}" ) - run_tests(test_configuration) + function_to_run(test_configuration) if __name__ == "__main__": - run_cli() + run_cli(run_tests) diff --git a/metricflow/test/populate_persistent_source_schemas.py b/metricflow/test/populate_persistent_source_schemas.py new file mode 100644 index 0000000000..0f0ff6f59c --- /dev/null +++ b/metricflow/test/populate_persistent_source_schemas.py @@ -0,0 +1,45 @@ +"""Script to help generate persistent source schemas with test data for all relevant engines.""" + +from __future__ import annotations + +import logging +import os + +from dbt_semantic_interfaces.enum_extension import assert_values_exhausted + +from metricflow.protocols.sql_client import SqlEngine +from metricflow.test.generate_snapshots import ( + MetricFlowTestConfiguration, + run_cli, + run_command, + set_engine_env_variables, +) + +logger = logging.getLogger(__name__) + + +def populate_schemas(test_configuration: MetricFlowTestConfiguration) -> None: # noqa: D + set_engine_env_variables(test_configuration) + + if test_configuration.engine is SqlEngine.DUCKDB or test_configuration.engine is SqlEngine.POSTGRES: + # DuckDB & Postgres don't use persistent source schema + return None + elif ( + test_configuration.engine is SqlEngine.SNOWFLAKE + or test_configuration.engine is SqlEngine.BIGQUERY + or test_configuration.engine is SqlEngine.DATABRICKS + or test_configuration.engine is SqlEngine.REDSHIFT + ): + engine_name = test_configuration.engine.value.lower() + os.environ["MF_TEST_ADAPTER_TYPE"] = engine_name + hatch_env = f"{engine_name}-env" + run_command( + f"hatch -v run {hatch_env}:pytest -vv --use-persistent-source-schema " + "metricflow/test/source_schema_tools.py::populate_source_schema" + ) + else: + assert_values_exhausted(test_configuration.engine) + + +if __name__ == "__main__": + run_cli(populate_schemas) From 9db0fe0d90b014178dc97f0cb09564a44a4c60da Mon Sep 17 00:00:00 2001 From: Devon Fulcher Date: Tue, 7 Nov 2023 18:52:43 -0600 Subject: [PATCH 2/2] Removed DatePart Enum to depend on DSI version instead (#853) Removed DatePart Enum and change imports to depend on DSI version instead. --- .../Under the Hood-20231107-184138.yaml | 6 ++ metricflow/dataset/convert_semantic_model.py | 2 +- metricflow/dataset/dataset.py | 2 +- metricflow/naming/linkable_spec_name.py | 3 +- .../plan_conversion/instance_converters.py | 2 +- metricflow/protocols/query_parameter.py | 3 +- metricflow/query/query_parser.py | 2 +- .../specs/query_param_implementations.py | 2 +- metricflow/specs/specs.py | 2 +- metricflow/sql/render/big_query.py | 2 +- metricflow/sql/render/databricks.py | 2 +- metricflow/sql/render/expr_renderer.py | 2 +- metricflow/sql/render/redshift.py | 2 +- metricflow/sql/render/snowflake.py | 2 +- metricflow/sql/sql_exprs.py | 2 +- .../test/integration/test_configured_cases.py | 2 +- metricflow/test/query/test_query_parser.py | 2 +- .../test_granularity_date_part_rendering.py | 2 +- metricflow/test/sql/test_sql_expr_render.py | 2 +- .../sql_clients/test_date_time_operations.py | 2 +- metricflow/time/date_part.py | 60 ------------------- metricflow/time/time_granularity_solver.py | 2 +- 22 files changed, 26 insertions(+), 82 deletions(-) create mode 100644 .changes/unreleased/Under the Hood-20231107-184138.yaml delete mode 100644 metricflow/time/date_part.py diff --git a/.changes/unreleased/Under the Hood-20231107-184138.yaml b/.changes/unreleased/Under the Hood-20231107-184138.yaml new file mode 100644 index 0000000000..51ff6aeb11 --- /dev/null +++ b/.changes/unreleased/Under the Hood-20231107-184138.yaml @@ -0,0 +1,6 @@ +kind: Under the Hood +body: Removed DatePart Enum and change imports to depend on DSI version instead. +time: 2023-11-07T18:41:38.606807-06:00 +custom: + Author: DevonFulcher + Issue: None diff --git a/metricflow/dataset/convert_semantic_model.py b/metricflow/dataset/convert_semantic_model.py index e435e69c22..cc29f45bca 100644 --- a/metricflow/dataset/convert_semantic_model.py +++ b/metricflow/dataset/convert_semantic_model.py @@ -10,6 +10,7 @@ from dbt_semantic_interfaces.protocols.measure import Measure from dbt_semantic_interfaces.protocols.semantic_model import SemanticModel from dbt_semantic_interfaces.references import SemanticModelElementReference, SemanticModelReference +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.aggregation_properties import AggregationState @@ -47,7 +48,6 @@ SqlSelectStatementNode, SqlTableFromClauseNode, ) -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/dataset/dataset.py b/metricflow/dataset/dataset.py index ea88d69b72..b32bf28274 100644 --- a/metricflow/dataset/dataset.py +++ b/metricflow/dataset/dataset.py @@ -4,12 +4,12 @@ from typing import Optional, Sequence from dbt_semantic_interfaces.references import TimeDimensionReference +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from dbt_semantic_interfaces.validations.unique_valid_name import MetricFlowReservedKeywords from metricflow.instances import InstanceSet, TimeDimensionInstance from metricflow.specs.specs import TimeDimensionSpec -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/naming/linkable_spec_name.py b/metricflow/naming/linkable_spec_name.py index 27fa92416b..bfcd1d6d12 100644 --- a/metricflow/naming/linkable_spec_name.py +++ b/metricflow/naming/linkable_spec_name.py @@ -4,10 +4,9 @@ from dataclasses import dataclass from typing import Optional, Tuple +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity -from metricflow.time.date_part import DatePart - DUNDER = "__" logger = logging.getLogger(__name__) diff --git a/metricflow/plan_conversion/instance_converters.py b/metricflow/plan_conversion/instance_converters.py index 07bf4922f1..fbe7d6c30a 100644 --- a/metricflow/plan_conversion/instance_converters.py +++ b/metricflow/plan_conversion/instance_converters.py @@ -9,6 +9,7 @@ from typing import Dict, List, Optional, Sequence, Tuple from dbt_semantic_interfaces.references import SemanticModelReference +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from more_itertools import bucket @@ -47,7 +48,6 @@ SqlFunctionExpression, ) from metricflow.sql.sql_plan import SqlSelectColumn -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/protocols/query_parameter.py b/metricflow/protocols/query_parameter.py index 0424b3f5a1..9285270e75 100644 --- a/metricflow/protocols/query_parameter.py +++ b/metricflow/protocols/query_parameter.py @@ -3,8 +3,7 @@ from typing import Optional, Protocol, Union, runtime_checkable from dbt_semantic_interfaces.type_enums import TimeGranularity - -from metricflow.time.date_part import DatePart +from dbt_semantic_interfaces.type_enums.date_part import DatePart @runtime_checkable diff --git a/metricflow/query/query_parser.py b/metricflow/query/query_parser.py index f302a48365..dbec189e57 100644 --- a/metricflow/query/query_parser.py +++ b/metricflow/query/query_parser.py @@ -21,6 +21,7 @@ MetricReference, TimeDimensionReference, ) +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.assert_one_arg import assert_exactly_one_arg_set @@ -55,7 +56,6 @@ WhereFilterSpec, ) from metricflow.specs.where_filter_transform import WhereSpecFactory -from metricflow.time.date_part import DatePart from metricflow.time.time_granularity_solver import ( PartialTimeDimensionSpec, RequestTimeGranularityException, diff --git a/metricflow/specs/query_param_implementations.py b/metricflow/specs/query_param_implementations.py index bf445f1435..96372c2d79 100644 --- a/metricflow/specs/query_param_implementations.py +++ b/metricflow/specs/query_param_implementations.py @@ -4,13 +4,13 @@ from typing import Optional from dbt_semantic_interfaces.protocols import ProtocolHint +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from typing_extensions import override from metricflow.naming.linkable_spec_name import StructuredLinkableSpecName from metricflow.protocols.query_parameter import InputOrderByParameter from metricflow.protocols.query_parameter import SavedQueryParameter as SavedQueryParameterProtocol -from metricflow.time.date_part import DatePart @dataclass(frozen=True) diff --git a/metricflow/specs/specs.py b/metricflow/specs/specs.py index 2b2e8b9544..601c4d0da0 100644 --- a/metricflow/specs/specs.py +++ b/metricflow/specs/specs.py @@ -26,6 +26,7 @@ TimeDimensionReference, ) from dbt_semantic_interfaces.type_enums.aggregation_type import AggregationType +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.aggregation_properties import AggregationState @@ -33,7 +34,6 @@ from metricflow.naming.linkable_spec_name import StructuredLinkableSpecName from metricflow.sql.sql_bind_parameters import SqlBindParameters from metricflow.sql.sql_column_type import SqlColumnType -from metricflow.time.date_part import DatePart from metricflow.visitor import VisitorOutputT diff --git a/metricflow/sql/render/big_query.py b/metricflow/sql/render/big_query.py index 83a4f82c38..7b78bad9ed 100644 --- a/metricflow/sql/render/big_query.py +++ b/metricflow/sql/render/big_query.py @@ -4,6 +4,7 @@ from typing import Collection from dbt_semantic_interfaces.enum_extension import assert_values_exhausted +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from typing_extensions import override @@ -25,7 +26,6 @@ SqlSubtractTimeIntervalExpression, ) from metricflow.sql.sql_plan import SqlSelectColumn -from metricflow.time.date_part import DatePart class BigQuerySqlExpressionRenderer(DefaultSqlExpressionRenderer): diff --git a/metricflow/sql/render/databricks.py b/metricflow/sql/render/databricks.py index bdd2b58d33..0a9d2de899 100644 --- a/metricflow/sql/render/databricks.py +++ b/metricflow/sql/render/databricks.py @@ -3,6 +3,7 @@ from typing import Collection from dbt_semantic_interfaces.enum_extension import assert_values_exhausted +from dbt_semantic_interfaces.type_enums.date_part import DatePart from typing_extensions import override from metricflow.errors.errors import UnsupportedEngineFeatureError @@ -13,7 +14,6 @@ ) from metricflow.sql.render.sql_plan_renderer import DefaultSqlQueryPlanRenderer from metricflow.sql.sql_exprs import SqlPercentileExpression, SqlPercentileFunctionType -from metricflow.time.date_part import DatePart class DatabricksSqlExpressionRenderer(DefaultSqlExpressionRenderer): diff --git a/metricflow/sql/render/expr_renderer.py b/metricflow/sql/render/expr_renderer.py index 8c603ffdc4..29fac9b1a1 100644 --- a/metricflow/sql/render/expr_renderer.py +++ b/metricflow/sql/render/expr_renderer.py @@ -8,6 +8,7 @@ from typing import Collection, List import jinja2 +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from typing_extensions import override @@ -37,7 +38,6 @@ SqlWindowFunctionExpression, ) from metricflow.sql.sql_plan import SqlSelectColumn -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/sql/render/redshift.py b/metricflow/sql/render/redshift.py index caa7398a3a..251227c209 100644 --- a/metricflow/sql/render/redshift.py +++ b/metricflow/sql/render/redshift.py @@ -3,6 +3,7 @@ from typing import Collection from dbt_semantic_interfaces.enum_extension import assert_values_exhausted +from dbt_semantic_interfaces.type_enums.date_part import DatePart from typing_extensions import override from metricflow.errors.errors import UnsupportedEngineFeatureError @@ -19,7 +20,6 @@ SqlPercentileExpression, SqlPercentileFunctionType, ) -from metricflow.time.date_part import DatePart class RedshiftSqlExpressionRenderer(DefaultSqlExpressionRenderer): diff --git a/metricflow/sql/render/snowflake.py b/metricflow/sql/render/snowflake.py index c368cbdf57..c6328feefe 100644 --- a/metricflow/sql/render/snowflake.py +++ b/metricflow/sql/render/snowflake.py @@ -3,6 +3,7 @@ from typing import Collection from dbt_semantic_interfaces.enum_extension import assert_values_exhausted +from dbt_semantic_interfaces.type_enums.date_part import DatePart from typing_extensions import override from metricflow.errors.errors import UnsupportedEngineFeatureError @@ -14,7 +15,6 @@ from metricflow.sql.render.sql_plan_renderer import DefaultSqlQueryPlanRenderer from metricflow.sql.sql_bind_parameters import SqlBindParameters from metricflow.sql.sql_exprs import SqlGenerateUuidExpression, SqlPercentileExpression, SqlPercentileFunctionType -from metricflow.time.date_part import DatePart class SnowflakeSqlExpressionRenderer(DefaultSqlExpressionRenderer): diff --git a/metricflow/sql/sql_exprs.py b/metricflow/sql/sql_exprs.py index 2206691e7c..c517c1bae1 100644 --- a/metricflow/sql/sql_exprs.py +++ b/metricflow/sql/sql_exprs.py @@ -12,6 +12,7 @@ from dbt_semantic_interfaces.enum_extension import assert_values_exhausted from dbt_semantic_interfaces.protocols.measure import MeasureAggregationParameters from dbt_semantic_interfaces.type_enums.aggregation_type import AggregationType +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.dag.id_generation import ( @@ -35,7 +36,6 @@ ) from metricflow.dag.mf_dag import DagNode, DisplayedProperty, NodeId from metricflow.sql.sql_bind_parameters import SqlBindParameters -from metricflow.time.date_part import DatePart from metricflow.visitor import Visitable, VisitorOutputT diff --git a/metricflow/test/integration/test_configured_cases.py b/metricflow/test/integration/test_configured_cases.py index 95579a11d1..de7faaeb4a 100644 --- a/metricflow/test/integration/test_configured_cases.py +++ b/metricflow/test/integration/test_configured_cases.py @@ -11,6 +11,7 @@ from dbt_semantic_interfaces.enum_extension import assert_values_exhausted from dbt_semantic_interfaces.implementations.elements.measure import PydanticMeasureAggregationParameters from dbt_semantic_interfaces.test_utils import as_datetime +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.engine.metricflow_engine import MetricFlowEngine, MetricFlowQueryRequest @@ -43,7 +44,6 @@ from metricflow.test.time.configurable_time_source import ( ConfigurableTimeSource, ) -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/test/query/test_query_parser.py b/metricflow/test/query/test_query_parser.py index 03e5ab9ccb..9a0815707f 100644 --- a/metricflow/test/query/test_query_parser.py +++ b/metricflow/test/query/test_query_parser.py @@ -8,6 +8,7 @@ from dbt_semantic_interfaces.parsing.objects import YamlConfigFile from dbt_semantic_interfaces.references import EntityReference from dbt_semantic_interfaces.test_utils import as_datetime +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.errors.errors import UnableToSatisfyQueryError @@ -30,7 +31,6 @@ from metricflow.test.fixtures.model_fixtures import query_parser_from_yaml from metricflow.test.model.example_project_configuration import EXAMPLE_PROJECT_CONFIGURATION_YAML_CONFIG_FILE from metricflow.test.time.metric_time_dimension import MTD -from metricflow.time.date_part import DatePart from metricflow.time.time_granularity_solver import RequestTimeGranularityException logger = logging.getLogger(__name__) diff --git a/metricflow/test/query_rendering/test_granularity_date_part_rendering.py b/metricflow/test/query_rendering/test_granularity_date_part_rendering.py index 548d33b088..38d2cb5c8e 100644 --- a/metricflow/test/query_rendering/test_granularity_date_part_rendering.py +++ b/metricflow/test/query_rendering/test_granularity_date_part_rendering.py @@ -8,6 +8,7 @@ import pytest from _pytest.fixtures import FixtureRequest +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.dataflow.builder.dataflow_plan_builder import DataflowPlanBuilder @@ -20,7 +21,6 @@ ) from metricflow.test.fixtures.setup_fixtures import MetricFlowTestSessionState from metricflow.test.query_rendering.compare_rendered_query import convert_and_check -from metricflow.time.date_part import DatePart @pytest.mark.sql_engine_snapshot diff --git a/metricflow/test/sql/test_sql_expr_render.py b/metricflow/test/sql/test_sql_expr_render.py index 5d3e309b36..c883313a4d 100644 --- a/metricflow/test/sql/test_sql_expr_render.py +++ b/metricflow/test/sql/test_sql_expr_render.py @@ -4,6 +4,7 @@ import textwrap import pytest +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.sql.render.expr_renderer import DefaultSqlExpressionRenderer @@ -30,7 +31,6 @@ SqlWindowFunctionExpression, SqlWindowOrderByArgument, ) -from metricflow.time.date_part import DatePart logger = logging.getLogger(__name__) diff --git a/metricflow/test/sql_clients/test_date_time_operations.py b/metricflow/test/sql_clients/test_date_time_operations.py index c0f0a7689e..60d77f6bae 100644 --- a/metricflow/test/sql_clients/test_date_time_operations.py +++ b/metricflow/test/sql_clients/test_date_time_operations.py @@ -21,6 +21,7 @@ import pandas as pd import pytest +from dbt_semantic_interfaces.type_enums.date_part import DatePart from metricflow.protocols.sql_client import SqlClient from metricflow.sql.sql_exprs import ( @@ -29,7 +30,6 @@ SqlExtractExpression, SqlStringLiteralExpression, ) -from metricflow.time.date_part import DatePart from metricflow.time.time_granularity import TimeGranularity logger = logging.getLogger(__name__) diff --git a/metricflow/time/date_part.py b/metricflow/time/date_part.py deleted file mode 100644 index cc1b5768d1..0000000000 --- a/metricflow/time/date_part.py +++ /dev/null @@ -1,60 +0,0 @@ -from __future__ import annotations - -from enum import Enum -from typing import List - -from dbt_semantic_interfaces.enum_extension import assert_values_exhausted -from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity - - -class DatePart(Enum): - """Date parts able to be extracted from a time dimension. - - Note this does not support WEEK (aka WEEKOFYEAR), because week numbering is very strange. - The ISO spec calls for weeks to start on Monday. Fair enough. It also calls for years to - start on Monday, but only about 1 out of every 7 do. In order to ensure years start on - Monday, the ISO decided that the first day of any given year is the Monday of the week - containing the first Thursday of that year. Consequently, the ISO standard produces - weeks numbered 1-53, but any days belonging to the preceding calendar year but in the - first week of the new year are part of the new ISO year. This is not really what people - expect. - - But there's more - different SQL engines also have different implementations of week of year. - When not using ISO, you get either 0-53, 1-54, or 1-53 with different ways of deciding - how to count the first few days in any given year. As such, we just don't support this. - - When the time comes, we can support week using whatever standard makes the most sense for - our usage context, but as it is not clear what that standard looks like we simply don't - support date_part = week for now. - - TODO: add support for hour, minute, second once those granularities are available - """ - - YEAR = "year" - QUARTER = "quarter" - MONTH = "month" - DAY = "day" - DOW = "dow" - DOY = "doy" - - def to_int(self) -> int: - """Convert to an int so that the size of the granularity can be easily compared.""" - if self is DatePart.DAY: - return TimeGranularity.DAY.to_int() - elif self is DatePart.DOW: - return TimeGranularity.DAY.to_int() - elif self is DatePart.DOY: - return TimeGranularity.DAY.to_int() - elif self is DatePart.MONTH: - return TimeGranularity.MONTH.to_int() - elif self is DatePart.QUARTER: - return TimeGranularity.QUARTER.to_int() - elif self is DatePart.YEAR: - return TimeGranularity.YEAR.to_int() - else: - assert_values_exhausted(self) - - @property - def compatible_granularities(self) -> List[TimeGranularity]: - """Granularities that can be queried with this date part.""" - return [granularity for granularity in TimeGranularity if granularity.to_int() >= self.to_int()] diff --git a/metricflow/time/time_granularity_solver.py b/metricflow/time/time_granularity_solver.py index 802fc7fd45..55e33ca865 100644 --- a/metricflow/time/time_granularity_solver.py +++ b/metricflow/time/time_granularity_solver.py @@ -13,6 +13,7 @@ MetricReference, TimeDimensionReference, ) +from dbt_semantic_interfaces.type_enums.date_part import DatePart from dbt_semantic_interfaces.type_enums.time_granularity import TimeGranularity from metricflow.dataflow.builder.node_data_set import DataflowPlanNodeOutputDataSetResolver @@ -23,7 +24,6 @@ from metricflow.specs.specs import ( TimeDimensionSpec, ) -from metricflow.time.date_part import DatePart from metricflow.time.time_granularity import ( adjust_to_end_of_period, adjust_to_start_of_period,