Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added flag to allow override SnowflakeDialect div_is_floor_div flag #545

Merged
merged 4 commits into from
Jan 9, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions DESCRIPTION.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,10 @@ Source code is also available at:
# Release Notes
- (Unreleased)
- Fix return value of snowflake get_table_names
- Added `force_div_is_floordiv` flag to override `div_is_floordiv` new default value `False` in `SnowflakeDialect`.
- With the flag in `False`, the `/` division operator will be treated as a float division and `//` as a floor division.
Copy link

@jochenott jochenott Nov 22, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is not how it works.

This flag is a signal from the dialect towards sqlalchemy that describes the behavior of this sql implementation (Snowflake). So if select 4/5 returns 0, div_is_floordiv must be set to True. If it returns 0.8, this flag must be set to False.

So the previous value of div_is_floordiv = True for snowflake was just a bug, which should be fixed. Just as for other bugs, there is no need to make this configurable.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @jochenott

We agree that the correct value for the flag div_is_floordiv in Snowflake should be False. However, current users expect the / and // operators to work as true divisions, although / should behave as a true division and // as a floor division. That's why the behavioral change will be introduced as a flag, which will be removed in a future major release.

Thank you for pointing out this code issue and your contribution!!!

- This flag is added to maintain backward compatibility with the previous behavior of Snowflake Dialect division.
- This flag will be removed in the future and Snowflake Dialect will use `div_is_floor_div` as `False`.

- v1.7.2(December 18, 2024)
- Fix quoting of `_` as column name
Expand Down
6 changes: 6 additions & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,11 @@ extra-dependencies = ["SQLAlchemy>=1.4.19,<2.0.0"]
features = ["development", "pandas"]
python = "3.8"

[tool.hatch.envs.sa14.scripts]
test-dialect = "pytest --ignore_v20_test -ra -vvv --tb=short --cov snowflake.sqlalchemy --cov-append --junitxml ./junit.xml --ignore=tests/sqlalchemy_test_suite tests/"
test-dialect-compatibility = "pytest --ignore_v20_test -ra -vvv --tb=short --cov snowflake.sqlalchemy --cov-append --junitxml ./junit.xml tests/sqlalchemy_test_suite"
test-dialect-aws = "pytest --ignore_v20_test -m \"aws\" -ra -vvv --tb=short --cov snowflake.sqlalchemy --cov-append --junitxml ./junit.xml --ignore=tests/sqlalchemy_test_suite tests/"

[tool.hatch.envs.default.env-vars]
COVERAGE_FILE = "coverage.xml"
SQLACHEMY_WARN_20 = "1"
Expand Down Expand Up @@ -134,4 +139,5 @@ markers = [
"requires_external_volume: tests that needs a external volume to be executed",
"external: tests that could but should only run on our external CI",
"feature_max_lob_size: tests that could but should only run on our external CI",
"feature_v20: tests that could but should only run on SqlAlchemy v20",
]
21 changes: 21 additions & 0 deletions src/snowflake/sqlalchemy/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
import operator
import re
import string
import warnings
from typing import List

from sqlalchemy import exc as sa_exc
Expand Down Expand Up @@ -802,6 +803,26 @@ def visit_join(self, join, asfrom=False, from_linter=None, **kwargs):
+ join.onclause._compiler_dispatch(self, from_linter=from_linter, **kwargs)
)

def visit_truediv_binary(self, binary, operator, **kw):
if self.dialect.div_is_floordiv:
warnings.warn(
"div_is_floordiv value will be changed to False in a future release. This will generate a behavior change on true and floor division. Please review https://docs.sqlalchemy.org/en/20/changelog/whatsnew_20.html#python-division-operator-performs-true-division-for-all-backends-added-floor-division",
PendingDeprecationWarning,
stacklevel=2,
)
return (
self.process(binary.left, **kw) + " / " + self.process(binary.right, **kw)
)

def visit_floordiv_binary(self, binary, operator, **kw):
if self.dialect.div_is_floordiv and IS_VERSION_20:
warnings.warn(
"div_is_floordiv value will be changed to False in a future release. This will generate a behavior change on true and floor division. Please review https://docs.sqlalchemy.org/en/20/changelog/whatsnew_20.html#python-division-operator-performs-true-division-for-all-backends-added-floor-division",
PendingDeprecationWarning,
stacklevel=2,
)
return super().visit_floordiv_binary(binary, operator, **kw)

def render_literal_value(self, value, type_):
# escape backslash
return super().render_literal_value(value, type_).replace("\\", "\\\\")
Expand Down
10 changes: 10 additions & 0 deletions src/snowflake/sqlalchemy/snowdialect.py
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,9 @@ class SnowflakeDialect(default.DefaultDialect):
colspecs = colspecs
ischema_names = ischema_names

# target database treats the / division operator as “floor division”
div_is_floordiv = False

# all str types must be converted in Unicode
convert_unicode = True

Expand Down Expand Up @@ -147,10 +150,17 @@ class SnowflakeDialect(default.DefaultDialect):

def __init__(
self,
force_div_is_floordiv: bool = True,
isolation_level: Optional[str] = SnowflakeIsolationLevel.READ_COMMITTED.value,
**kwargs: Any,
):
super().__init__(isolation_level=isolation_level, **kwargs)
self.force_div_is_floordiv = force_div_is_floordiv
self.div_is_floordiv = force_div_is_floordiv

def initialize(self, connection):
super().initialize(connection)
self.div_is_floordiv = self.force_div_is_floordiv

@classmethod
def dbapi(cls):
Expand Down
20 changes: 20 additions & 0 deletions tests/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,26 @@
TEST_SCHEMA = f"sqlalchemy_tests_{str(uuid.uuid4()).replace('-', '_')}"


def pytest_addoption(parser):
parser.addoption(
"--ignore_v20_test",
action="store_true",
default=False,
help="skip sqlalchemy 2.0 exclusive tests",
)


def pytest_collection_modifyitems(config, items):
if config.getoption("--ignore_v20_test"):
# --ignore_v20_test given in cli: skip sqlalchemy 2.0 tests
skip_feature_v2 = pytest.mark.skip(
reason="need remove --ignore_v20_test option to run"
)
for item in items:
if "feature_v20" in item.keywords:
item.add_marker(skip_feature_v2)


@pytest.fixture(scope="session")
def on_travis():
return os.getenv("TRAVIS", "").lower() == "true"
Expand Down
79 changes: 79 additions & 0 deletions tests/test_compiler.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,14 @@
# Copyright (c) 2012-2023 Snowflake Computing Inc. All rights reserved.
#

import pytest
from sqlalchemy import Integer, String, and_, func, insert, select
from sqlalchemy.schema import DropColumnComment, DropTableComment
from sqlalchemy.sql import column, quoted_name, table
from sqlalchemy.testing.assertions import AssertsCompiledSQL

from snowflake.sqlalchemy import snowdialect
from src.snowflake.sqlalchemy.snowdialect import SnowflakeDialect

table1 = table(
"table1", column("id", Integer), column("name", String), column("value", Integer)
Expand Down Expand Up @@ -135,3 +137,80 @@ def test_outer_lateral_join():
str(stmt.compile(dialect=snowdialect.dialect()))
== "SELECT colname AS label \nFROM abc JOIN LATERAL flatten(PARSE_JSON(colname2)) AS anon_1 GROUP BY colname"
)


@pytest.mark.feature_v20
def test_division_operator_with_force_div_is_floordiv_false():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 / col2
assert (
str(stmt.compile(dialect=SnowflakeDialect(force_div_is_floordiv=False)))
== "col1 / col2"
)


@pytest.mark.feature_v20
def test_division_operator_with_denominator_expr_force_div_is_floordiv_false():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 / func.sqrt(col2)
assert (
str(stmt.compile(dialect=SnowflakeDialect(force_div_is_floordiv=False)))
== "col1 / sqrt(col2)"
)


@pytest.mark.feature_v20
def test_division_operator_with_force_div_is_floordiv_default_true():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 / col2
assert str(stmt.compile(dialect=SnowflakeDialect())) == "col1 / col2"


@pytest.mark.feature_v20
def test_division_operator_with_denominator_expr_force_div_is_floordiv_default_true():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 / func.sqrt(col2)
assert str(stmt.compile(dialect=SnowflakeDialect())) == "col1 / sqrt(col2)"


@pytest.mark.feature_v20
def test_floor_division_operator_force_div_is_floordiv_false():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 // col2
assert (
str(stmt.compile(dialect=SnowflakeDialect(force_div_is_floordiv=False)))
== "FLOOR(col1 / col2)"
)


@pytest.mark.feature_v20
def test_floor_division_operator_with_denominator_expr_force_div_is_floordiv_false():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 // func.sqrt(col2)
assert (
str(stmt.compile(dialect=SnowflakeDialect(force_div_is_floordiv=False)))
== "FLOOR(col1 / sqrt(col2))"
)


@pytest.mark.feature_v20
def test_floor_division_operator_force_div_is_floordiv_default_true():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 // col2
assert str(stmt.compile(dialect=SnowflakeDialect())) == "col1 / col2"


@pytest.mark.feature_v20
def test_floor_division_operator_with_denominator_expr_force_div_is_floordiv_default_true():
col1 = column("col1", Integer)
col2 = column("col2", Integer)
stmt = col1 // func.sqrt(col2)
res = stmt.compile(dialect=SnowflakeDialect())
assert str(res) == "FLOOR(col1 / sqrt(col2))"
87 changes: 86 additions & 1 deletion tests/test_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,13 +31,15 @@
create_engine,
dialects,
exc,
func,
insert,
inspect,
text,
)
from sqlalchemy.exc import DBAPIError, NoSuchTableError, OperationalError
from sqlalchemy.sql import and_, not_, or_, select
from sqlalchemy.sql import and_, literal, not_, or_, select
from sqlalchemy.sql.ddl import CreateTable
from sqlalchemy.testing.assertions import eq_

import snowflake.connector.errors
import snowflake.sqlalchemy.snowdialect
Expand Down Expand Up @@ -1864,3 +1866,86 @@ def test_snowflake_sqlalchemy_as_valid_client_type():
snowflake.connector.connection.DEFAULT_CONFIGURATION[
"internal_application_version"
] = origin_internal_app_version


@pytest.mark.parametrize(
"operation",
[
[
literal(5),
literal(10),
0.5,
],
[literal(5), func.sqrt(literal(10)), 1.5811388300841895],
[literal(4), literal(5), decimal.Decimal("0.800000")],
[literal(2), literal(2), 1.0],
[literal(3), literal(2), 1.5],
[literal(4), literal(1.5), 2.666667],
[literal(5.5), literal(10.7), 0.5140187],
[literal(5.5), literal(8), 0.6875],
],
)
def test_true_division_operation(engine_testaccount, operation):
# expected_warning = "div_is_floordiv value will be changed to False in a future release. This will generate a behavior change on true and floor division. Please review https://docs.sqlalchemy.org/en/20/changelog/whatsnew_20.html#python-division-operator-performs-true-division-for-all-backends-added-floor-division"
# with pytest.warns(PendingDeprecationWarning, match=expected_warning):
with engine_testaccount.connect() as conn:
eq_(
conn.execute(select(operation[0] / operation[1])).fetchall(),
[((operation[2]),)],
)


@pytest.mark.parametrize(
"operation",
[
[literal(5), literal(10), 0.5, 0.5],
[literal(5), func.sqrt(literal(10)), 1.5811388300841895, 1.0],
[
literal(4),
literal(5),
decimal.Decimal("0.800000"),
decimal.Decimal("0.800000"),
],
[literal(2), literal(2), 1.0, 1.0],
[literal(3), literal(2), 1.5, 1.5],
[literal(4), literal(1.5), 2.666667, 2.0],
[literal(5.5), literal(10.7), 0.5140187, 0],
[literal(5.5), literal(8), 0.6875, 0.6875],
],
)
@pytest.mark.feature_v20
def test_division_force_div_is_floordiv_default(engine_testaccount, operation):
expected_warning = "div_is_floordiv value will be changed to False in a future release. This will generate a behavior change on true and floor division. Please review https://docs.sqlalchemy.org/en/20/changelog/whatsnew_20.html#python-division-operator-performs-true-division-for-all-backends-added-floor-division"
with pytest.warns(PendingDeprecationWarning, match=expected_warning):
with engine_testaccount.connect() as conn:
eq_(
conn.execute(
select(operation[0] / operation[1], operation[0] // operation[1])
).fetchall(),
[(operation[2], operation[3])],
)


@pytest.mark.parametrize(
"operation",
[
[literal(5), literal(10), 0.5, 0],
[literal(5), func.sqrt(literal(10)), 1.5811388300841895, 1.0],
[literal(4), literal(5), decimal.Decimal("0.800000"), 0],
[literal(2), literal(2), 1.0, 1.0],
[literal(3), literal(2), 1.5, 1],
[literal(4), literal(1.5), 2.666667, 2.0],
[literal(5.5), literal(10.7), 0.5140187, 0],
[literal(5.5), literal(8), 0.6875, 0],
],
)
@pytest.mark.feature_v20
def test_division_force_div_is_floordiv_false(db_parameters, operation):
engine = create_engine(URL(**db_parameters), **{"force_div_is_floordiv": False})
with engine.connect() as conn:
eq_(
conn.execute(
select(operation[0] / operation[1], operation[0] // operation[1])
).fetchall(),
[(operation[2], operation[3])],
)
Loading