Skip to content

Commit

Permalink
Add mixins (#1484)
Browse files Browse the repository at this point in the history
* Basic solution

SNOW-1636849 Auto-teardown Native App in integration tests (#1478)

Changes `with project_directory()` to `with nativeapp_project_directory()`, which automatically runs `snow app teardown` before exiting the project. This allows us to remove the `try`/`finally` in most tests. For tests that were using `with pushd(test_project)`, this has been changed to `with nativeapp_teardown()`, which is what `with nativeapp_project_directory()` uses under the hood.

SNOW-1621834 Cast version to identifier when creating/dropping app versions (#1475)

When running `snow app version create` and `snow app version drop`, wrap the version in `to_identifier()` so users don't have to specify the quotes around version names that aren't valid identifiers. If the name is already quoted, `to_identifier()` doesn't do anything.

Added tests

Added tests

* Added tests

Added tests

* Post-review-fixes
  • Loading branch information
sfc-gh-jsikorski authored and sfc-gh-jvasquezrojas committed Aug 26, 2024
1 parent 9aa0209 commit 56af21a
Show file tree
Hide file tree
Showing 13 changed files with 281 additions and 9 deletions.
15 changes: 14 additions & 1 deletion src/snowflake/cli/api/project/schemas/entities/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
from abc import ABC
from typing import Generic, List, Optional, TypeVar, Union

from pydantic import Field, PrivateAttr
from pydantic import Field, PrivateAttr, field_validator
from snowflake.cli.api.identifiers import FQN
from snowflake.cli.api.project.schemas.identifier_model import Identifier
from snowflake.cli.api.project.schemas.native_app.application import (
Expand All @@ -41,6 +41,19 @@ class MetaField(UpdatableModel):
title="Actions that will be executed after the application object is created/upgraded",
default=None,
)
use_mixins: Optional[List[str]] = Field(
title="Name of the mixin used to fill the entity fields",
default=None,
)

@field_validator("use_mixins", mode="before")
@classmethod
def ensure_use_mixins_is_a_list(
cls, mixins: Optional[str | List[str]]
) -> Optional[List[str]]:
if isinstance(mixins, str):
return [mixins]
return mixins


class DefaultsField(UpdatableModel):
Expand Down
65 changes: 57 additions & 8 deletions src/snowflake/cli/api/project/schemas/project_definition.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@
from __future__ import annotations

from dataclasses import dataclass
from typing import Dict, Optional, Union
from typing import Any, Dict, List, Optional, Union

from packaging.version import Version
from pydantic import Field, ValidationError, field_validator, model_validator
Expand Down Expand Up @@ -123,15 +123,11 @@ def apply_defaults(cls, data: Dict) -> Dict:
"""
if "defaults" in data and "entities" in data:
for key, entity in data["entities"].items():
entity_type = entity["type"]
if entity_type not in v2_entity_model_types_map:
entity_fields = get_allowed_fields_for_entity(entity)
if not entity_fields:
continue
entity_model = v2_entity_model_types_map[entity_type]
for default_key, default_value in data["defaults"].items():
if (
default_key in entity_model.model_fields
and default_key not in entity
):
if default_key in entity_fields and default_key not in entity:
entity[default_key] = default_value
return data

Expand Down Expand Up @@ -194,6 +190,36 @@ def _validate_target_field(
default=None,
)

mixins: Optional[Dict[str, Dict]] = Field(
title="Mixins to apply to entities",
default=None,
)

@model_validator(mode="before")
@classmethod
def apply_mixins(cls, data: Dict) -> Dict:
"""
Applies mixins to those entities, whose meta field contains the mixin name.
"""
if "mixins" not in data or "entities" not in data:
return data

for entity in data["entities"].values():
entity_mixins = entity_mixins_to_list(
entity.get("meta", {}).get("use_mixins")
)

entity_fields = get_allowed_fields_for_entity(entity)
if entity_fields and entity_mixins:
for mixin_name in entity_mixins:
if mixin_name in data["mixins"]:
for key, value in data["mixins"][mixin_name].items():
if key in entity_fields:
entity[key] = value
else:
raise ValueError(f"Mixin {mixin_name} not found in mixins")
return data

def get_entities_by_type(self, entity_type: str):
return {i: e for i, e in self.entities.items() if e.get_type() == entity_type}

Expand Down Expand Up @@ -222,3 +248,26 @@ def get_version_map():
if FeatureFlag.ENABLE_PROJECT_DEFINITION_V2.is_enabled():
version_map["2"] = DefinitionV20
return version_map


def entity_mixins_to_list(entity_mixins: Optional[str | List[str]]) -> List[str]:
"""
Convert an optional string or a list of strings to a list of strings.
"""
if entity_mixins is None:
return []
if isinstance(entity_mixins, str):
return [entity_mixins]
return entity_mixins


def get_allowed_fields_for_entity(entity: Dict[str, Any]) -> List[str]:
"""
Get the allowed fields for the given entity.
"""
entity_type = entity.get("type")
if entity_type not in v2_entity_model_types_map:
return []

entity_model = v2_entity_model_types_map[entity_type]
return entity_model.model_fields
34 changes: 34 additions & 0 deletions tests/project/test_project_definition_v2.py
Original file line number Diff line number Diff line change
Expand Up @@ -385,6 +385,40 @@ def test_v1_to_v2_conversion(
_assert_entities_are_equal(v1_function, v2_function)


# TODO:
# 1. rewrite projects to have one big definition covering all complex positive cases
# 2. Add negative case - entity uses non-existent mixin
@pytest.mark.parametrize(
"project_name,stage1,stage2",
[("mixins_basic", "foo", "bar"), ("mixins_defaults_hierarchy", "foo", "baz")],
)
def test_mixins(project_directory, project_name, stage1, stage2):
with project_directory(project_name) as project_dir:
definition = DefinitionManager(project_dir).project_definition

assert definition.entities["function1"].stage == stage1
assert definition.entities["function1"].handler == "app.hello"
assert definition.entities["function2"].stage == stage2
assert definition.entities["function1"].handler == "app.hello"


def test_mixins_for_different_entities(project_directory):
with project_directory("mixins_different_entities") as project_dir:
definition = DefinitionManager(project_dir).project_definition

assert definition.entities["function1"].stage == "foo"
assert definition.entities["streamlit1"].main_file == "streamlit_app.py"


def test_list_of_mixins_in_correct_order(project_directory):
with project_directory("mixins_list_applied_in_order") as project_dir:
definition = DefinitionManager(project_dir).project_definition

assert definition.entities["function1"].stage == "foo"
assert definition.entities["function2"].stage == "baz"
assert definition.entities["streamlit1"].stage == "bar"


def _assert_entities_are_equal(
v1_entity: _CallableBase, v2_entity: SnowparkEntityModel
):
Expand Down
28 changes: 28 additions & 0 deletions tests/test_data/projects/mixins_basic/snowflake.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
definition_version: '2'
entities:
function1:
artifacts:
- src
handler: app.hello
identifier: name
meta:
use_mixins: my_mixin
returns: string
signature:
- name: name
type: string
type: function
function2:
artifacts:
- src
handler: app.hello
identifier: name
returns: string
signature:
- name: name
type: string
stage: bar
type: function
mixins:
my_mixin:
stage: foo
29 changes: 29 additions & 0 deletions tests/test_data/projects/mixins_defaults_hierarchy/snowflake.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
definition_version: '2'
entities:
function1:
artifacts:
- src
handler: app.hello
identifier: name
meta:
use_mixins: my_mixin
returns: string
signature:
- name: name
type: string
type: function
function2:
artifacts:
- src
handler: app.hello2
identifier: name
returns: string
signature:
- name: name
type: string
type: function
defaults:
stage: baz
mixins:
my_mixin:
stage: foo
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
name: sf_env
channels:
- snowflake
dependencies:
- pandas
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import streamlit as st

st.title("Example page")
45 changes: 45 additions & 0 deletions tests/test_data/projects/mixins_different_entities/snowflake.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
definition_version: '2'
entities:
function1:
artifacts:
- src
handler: app.hello
identifier: name
meta:
use_mixins: my_mixin
returns: string
signature:
- name: name
type: string
type: function
function2:
artifacts:
- src
handler: app.hello
identifier: name
returns: string
signature:
- name: name
type: string
type: function
streamlit1:
artifacts:
- streamlit_app.py
- environment.yml
- pages
identifier:
name: test_streamlit
pages_dir: non_existent_dir
query_warehouse: test_warehouse
stage: streamlit
title: My Fancy Streamlit
type: streamlit
meta:
use_mixins: my_mixin
defaults:
stage: baz
mixins:
my_mixin:
stage: foo
main_file: streamlit_app.py
pages_dir: pages
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
name: sf_env
channels:
- snowflake
dependencies:
- pandas
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import streamlit as st

st.title("Example page")
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
definition_version: '2'
entities:
function1:
artifacts:
- src
handler: app.hello
identifier: name
meta:
use_mixins:
- second_mixin
- first_mixin
returns: string
signature:
- name: name
type: string
type: function
function2:
artifacts:
- src
handler: app.hello
identifier: name
returns: string
meta:
use_mixins:
- third_mixin
signature:
- name: name
type: string
type: function
streamlit1:
artifacts:
- streamlit_app.py
- environment.yml
- pages
identifier:
name: test_streamlit
pages_dir: non_existent_dir
query_warehouse: test_warehouse
stage: streamlit
title: My Fancy Streamlit
type: streamlit
meta:
use_mixins:
- first_mixin
- second_mixin
mixins:
first_mixin:
stage: foo
main_file: streamlit_app.py
pages_dir: non_existent_pages_dir
second_mixin:
stage: bar
main_file: streamlit_app.py
pages_dir: pages
third_mixin:
stage: baz
main_file: streamlit_app.py
pages_dir: pages
Empty file.

0 comments on commit 56af21a

Please sign in to comment.