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

allows naming conventions to be changed #998

Merged
merged 114 commits into from
Jun 26, 2024
Merged
Show file tree
Hide file tree
Changes from 34 commits
Commits
Show all changes
114 commits
Select commit Hold shift + click to select a range
93995b9
allows to decorate async function with dlt.source
rudolfix Feb 20, 2024
4444878
adds pytest-async and updates pytest to 7.x
rudolfix Feb 20, 2024
b3b70f6
fixes forked teardown issue 7.x
rudolfix Feb 20, 2024
f5d7a0a
bumps deps for py 3.12
rudolfix Feb 21, 2024
83dc38a
adds py 12 common tests
rudolfix Feb 21, 2024
21ebfee
fixes typings after deps bump
rudolfix Feb 21, 2024
7985f9d
bumps airflow, yanks duckdb to 0.9.2
rudolfix Feb 21, 2024
07f285e
fixes tests
rudolfix Feb 21, 2024
06e441e
fixes pandas version
rudolfix Feb 21, 2024
3e846a1
adds 3.12 duckdb dep
rudolfix Feb 22, 2024
37b4a31
Merge branch 'devel' into rfix/enables-async-source
rudolfix Feb 22, 2024
934c167
Merge branch 'devel' into rfix/enables-async-source
rudolfix Feb 24, 2024
7fa574d
adds right hand pipe operator
rudolfix Feb 24, 2024
8c7942d
fixes docker ci build
rudolfix Feb 24, 2024
f951fc0
adds docs on async sources and resources
rudolfix Feb 24, 2024
387a7c7
normalizes default hints and preferred types in schema
rudolfix Feb 25, 2024
88728e1
defines pipeline state table in utils, column normalization in simple…
rudolfix Feb 25, 2024
1a53425
normalizes all identifiers used by relational normalizer, fixes other…
rudolfix Feb 25, 2024
8835023
fixes sql job client to use normalized identifiers in queries
rudolfix Feb 25, 2024
f4c504f
runs state sync tests for lower and upper case naming conventions
rudolfix Feb 25, 2024
874cc29
fixes weaviate to use normalized identifiers in queries
rudolfix Feb 25, 2024
c4e9f35
partially fixes qdrant incorrect state and version retrieval queries
rudolfix Feb 25, 2024
6345377
initial sql uppercase naming convention
rudolfix Feb 25, 2024
96a02ff
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Mar 8, 2024
aef8cc2
adds native df readers to databricks and bigquery
rudolfix Mar 9, 2024
a53c00b
adds casing identifier capability to support different casing in nami…
rudolfix Mar 9, 2024
91f5780
cleans typing for relational normalizer
rudolfix Mar 9, 2024
5984824
renames escape functions
rudolfix Mar 18, 2024
3458441
destination capabilities for case fold and case sensitivity
rudolfix Mar 18, 2024
55362b0
drops supports naming module and allows naming to be instance in conf…
rudolfix Mar 18, 2024
b836dfe
checks all tables in information schema in one go, observes case fold…
rudolfix Mar 18, 2024
e50bfaa
moves schema verification to destination utils
rudolfix Mar 18, 2024
42d149f
adds method to remove processing hints from schema, helper functions …
rudolfix Mar 18, 2024
c53808f
accepts naming convention instances when resolving configs
rudolfix Mar 18, 2024
b97ae53
fixes the cloning of schema in decorator, removes processing hints
rudolfix Mar 18, 2024
0132c2f
removes processing hints when saving imported schema
rudolfix Mar 18, 2024
2a7c5dd
adds docs on naming conventions, removes technical docs
rudolfix Mar 18, 2024
d502c7c
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Jun 4, 2024
3e7504b
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Jun 6, 2024
3bb929f
adds casing info to databrick caps, makes caps an instance attr
rudolfix Jun 11, 2024
9f0920c
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Jun 11, 2024
724dc15
adjusts destination casing in caps from schema naming and config
rudolfix Jun 11, 2024
b58a118
raises detailed schema identifier clash exceptions
rudolfix Jun 11, 2024
d190ea1
adds is_case_sensitive and name to NamingConvention
rudolfix Jun 11, 2024
b445654
adds sanity check if _dlt prefix is preserved
rudolfix Jun 11, 2024
ee8a95b
finds genric types in non generic classes deriving from generic
rudolfix Jun 11, 2024
eb30838
uses casefold INSERT VALUES job column names
rudolfix Jun 11, 2024
558db91
adds a method make_qualified_table_name_path that calculates componen…
rudolfix Jun 11, 2024
dea9669
adds casing info to destinations, caps as instance attrs, custom tabl…
rudolfix Jun 11, 2024
b1e2b09
adds naming convention to restore state tests, make them essential
rudolfix Jun 11, 2024
210be70
fixes table builder tests
rudolfix Jun 11, 2024
95b703d
removes processing hints when exporting schema to import folder, warn…
rudolfix Jun 12, 2024
4b72b77
allows to subclass INFO SCHEMA query generation and uses specialized …
rudolfix Jun 12, 2024
ab39e06
uses correct schema escaping function in sql jobs
rudolfix Jun 12, 2024
2ae3ad2
passes pipeline state to package state via extract
rudolfix Jun 12, 2024
09b7731
fixes optional normalizers module
rudolfix Jun 12, 2024
cfd3e5f
excludes version_hash from pipeline state SELECT
rudolfix Jun 12, 2024
0edbbfd
passes pipeline state to package state pt.2
rudolfix Jun 12, 2024
5769ba1
re-enables sentry tests
rudolfix Jun 12, 2024
1f17a44
bumps qdrant client, makes test running for local version
rudolfix Jun 12, 2024
71e418b
makes weaviate running
rudolfix Jun 12, 2024
ce414e1
uses schemata to find databases on athena
rudolfix Jun 13, 2024
bde61a9
uses api get_table for hidden dataset on bigquery to reflect schemas,…
rudolfix Jun 13, 2024
036e3dd
adds naming conventions to two restore state tests
rudolfix Jun 13, 2024
8546763
fixes escape identifiers to column escape
rudolfix Jun 13, 2024
f57e286
fix conflicts in docs
rudolfix Jun 13, 2024
cf50bd4
adjusts capabilities in capabilities() method, uses config and naming…
rudolfix Jun 15, 2024
72969ce
allows to add props to classes without vectorizer in weaviate
rudolfix Jun 15, 2024
656d5fc
moves caps function into factories, cleansup adapters and custom dest…
rudolfix Jun 15, 2024
bbd7fe6
sentry_dsn
rudolfix Jun 15, 2024
a671508
adds basic destination reference tests
rudolfix Jun 15, 2024
81e0db9
fixes table builder tests
rudolfix Jun 15, 2024
8a32793
fix deps and docs
rudolfix Jun 15, 2024
0dc6dc8
fixes more tests
rudolfix Jun 16, 2024
4a39795
case sensitivity docs stubs
rudolfix Jun 17, 2024
43d6d5f
fixes drop_pipeline fixture
rudolfix Jun 17, 2024
e3d998c
improves partial config generation for capabilities
rudolfix Jun 17, 2024
3aef3fd
adds snowflake csv support
rudolfix Jun 17, 2024
6df7a34
creates separate csv tests
rudolfix Jun 17, 2024
57aec2e
allows to import files into extract storage, adds import file writer …
rudolfix Jun 19, 2024
fee7af5
handles ImportFileMeta in extractor
rudolfix Jun 19, 2024
96c7222
adds import file item normalizer and router to normalize
rudolfix Jun 19, 2024
116add0
supports csv format config for snowflake
rudolfix Jun 19, 2024
42eacaf
removes realpath wherever possible and adds fast make_full_path to Fi…
rudolfix Jun 20, 2024
3793d06
adds additional methods to load_package storage to make listings faster
rudolfix Jun 20, 2024
88eec9c
adds file_format to dlt.resource, uses preferred file format for dlt …
rudolfix Jun 21, 2024
8e0f0a8
docs for importing files, file_format
rudolfix Jun 21, 2024
b1c095c
code improvements and tests
rudolfix Jun 21, 2024
46ec732
docs hard links note
rudolfix Jun 21, 2024
2194b18
Merge pull request #1479 from dlt-hub/feat/snowflake-csv-support
rudolfix Jun 21, 2024
1384ed3
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Jun 21, 2024
b00cbb2
moves loader parallelism test to pipeliens, solves duckdb ci test err…
rudolfix Jun 23, 2024
a530345
fixes tests
rudolfix Jun 23, 2024
4271895
moves drop_pipeline fixture level up
rudolfix Jun 23, 2024
abd02df
drops default naming convention from caps so naming in saved schema p…
rudolfix Jun 24, 2024
14b4b0e
unifies all representations of pipeline state
rudolfix Jun 24, 2024
60e45b1
tries to decompress text file first in fs_client
rudolfix Jun 24, 2024
a84be2a
tests get stored state in test_job_client
rudolfix Jun 24, 2024
1dc7a09
removes credentials from dlt.attach, addes destination and staging fa…
rudolfix Jun 24, 2024
ab69b76
cleans up env variables and pipeline dropping fixutere precedence
rudolfix Jun 24, 2024
0eeb21d
Merge branch 'devel' into rfix/allows-naming-conventions
rudolfix Jun 24, 2024
f1097d8
removes dev_mode from dlt.attach
rudolfix Jun 24, 2024
3855fcc
adds missing arguments to filesystem factory
rudolfix Jun 24, 2024
651412e
fixes tests
rudolfix Jun 24, 2024
aab36e1
updates destination and naming convention docs
rudolfix Jun 25, 2024
7294aae
removes is_case_sensitive from naming convention initializer
rudolfix Jun 26, 2024
dc10473
simplifies with_file_import mark
rudolfix Jun 26, 2024
727a35e
adds case sensitivity tests
rudolfix Jun 26, 2024
4cb2646
uses dev_mode everywhere
rudolfix Jun 26, 2024
f098e5a
improves csv docs
rudolfix Jun 26, 2024
1521778
fixes encodings in fsspec
rudolfix Jun 26, 2024
796483e
improves naming convention docs
rudolfix Jun 26, 2024
534c7f8
fixes tests and renames clash to collision
rudolfix Jun 26, 2024
5f4cb4c
fixes getting original bases from instance
rudolfix Jun 26, 2024
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
8 changes: 5 additions & 3 deletions dlt/common/configuration/specs/base_configuration.py
Original file line number Diff line number Diff line change
Expand Up @@ -197,11 +197,13 @@ def default_factory(att_value=att_value): # type: ignore[no-untyped-def]

@configspec
class BaseConfiguration(MutableMapping[str, Any]):
__is_resolved__: bool = dataclasses.field(default=False, init=False, repr=False)
__is_resolved__: bool = dataclasses.field(default=False, init=False, repr=False, compare=False)
"""True when all config fields were resolved and have a specified value type"""
__section__: str = dataclasses.field(default=None, init=False, repr=False)
__section__: str = dataclasses.field(default=None, init=False, repr=False, compare=False)
"""Obligatory section used by config providers when searching for keys, always present in the search path"""
__exception__: Exception = dataclasses.field(default=None, init=False, repr=False)
__exception__: Exception = dataclasses.field(
default=None, init=False, repr=False, compare=False
)
"""Holds the exception that prevented the full resolution"""
__config_gen_annotations__: ClassVar[List[str]] = []
"""Additional annotations for config generator, currently holds a list of fields of interest that have defaults"""
Expand Down
4 changes: 2 additions & 2 deletions dlt/common/data_writers/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
from dlt.common.data_writers.escape import (
escape_redshift_literal,
escape_redshift_identifier,
escape_bigquery_identifier,
escape_hive_identifier,
)

__all__ = [
Expand All @@ -14,5 +14,5 @@
"new_file_id",
"escape_redshift_literal",
"escape_redshift_identifier",
"escape_bigquery_identifier",
"escape_hive_identifier",
]
6 changes: 3 additions & 3 deletions dlt/common/data_writers/escape.py
Original file line number Diff line number Diff line change
Expand Up @@ -121,18 +121,18 @@ def escape_redshift_identifier(v: str) -> str:
escape_athena_identifier = escape_postgres_identifier


def escape_bigquery_identifier(v: str) -> str:
def escape_hive_identifier(v: str) -> str:
# https://cloud.google.com/bigquery/docs/reference/standard-sql/lexical
return "`" + v.replace("\\", "\\\\").replace("`", "\\`") + "`"


def escape_snowflake_identifier(v: str) -> str:
# Snowcase uppercase all identifiers unless quoted. Match this here so queries on information schema work without issue
# See also https://docs.snowflake.com/en/sql-reference/identifiers-syntax#double-quoted-identifiers
return escape_postgres_identifier(v.upper())
return escape_postgres_identifier(v)


escape_databricks_identifier = escape_bigquery_identifier
escape_databricks_identifier = escape_hive_identifier


DATABRICKS_ESCAPE_DICT = {"'": "\\'", "\\": "\\\\", "\n": "\\n", "\r": "\\r"}
Expand Down
20 changes: 14 additions & 6 deletions dlt/common/destination/capabilities.py
Original file line number Diff line number Diff line change
@@ -1,12 +1,10 @@
from typing import Any, Callable, ClassVar, List, Literal, Optional, Tuple, Set, get_args
from typing import Any, Callable, ClassVar, List, Literal, Optional, Tuple, Set, Union, get_args

from dlt.common.configuration.utils import serialize_value
from dlt.common.configuration import configspec
from dlt.common.configuration.specs import ContainerInjectableContext
from dlt.common.utils import identity

from dlt.common.arithmetics import DEFAULT_NUMERIC_PRECISION, DEFAULT_NUMERIC_SCALE

from dlt.common.normalizers.naming import NamingConvention
from dlt.common.wei import EVM_DECIMAL_PRECISION

# known loader file formats
Expand Down Expand Up @@ -35,7 +33,13 @@ class DestinationCapabilitiesContext(ContainerInjectableContext):
preferred_staging_file_format: Optional[TLoaderFileFormat]
supported_staging_file_formats: List[TLoaderFileFormat]
escape_identifier: Callable[[str], str]
"Escapes table name, column name and other identifiers"
escape_literal: Callable[[Any], Any]
"Escapes string literal"
casefold_identifier: Callable[[str], str] = str
"""Casing function applied by destination to represent case insensitive identifiers."""
has_case_sensitive_identifiers: bool
"""Tells if identifiers in destination are case sensitive, before case_identifier function is applied"""
decimal_precision: Tuple[int, int]
wei_precision: Tuple[int, int]
max_identifier_length: int
Expand All @@ -46,7 +50,7 @@ class DestinationCapabilitiesContext(ContainerInjectableContext):
is_max_text_data_type_length_in_bytes: bool
supports_transactions: bool
supports_ddl_transactions: bool
naming_convention: str = "snake_case"
naming_convention: Union[str, NamingConvention] = "snake_case"
alter_add_multi_column: bool = True
supports_truncate_command: bool = True
schema_supports_numeric_precision: bool = True
Expand All @@ -63,14 +67,18 @@ class DestinationCapabilitiesContext(ContainerInjectableContext):
@staticmethod
def generic_capabilities(
preferred_loader_file_format: TLoaderFileFormat = None,
naming_convention: Union[str, NamingConvention] = None,
) -> "DestinationCapabilitiesContext":
caps = DestinationCapabilitiesContext()
caps.preferred_loader_file_format = preferred_loader_file_format
caps.supported_loader_file_formats = ["jsonl", "insert_values", "parquet"]
caps.preferred_staging_file_format = None
caps.supported_staging_file_formats = []
caps.escape_identifier = identity
caps.naming_convention = naming_convention or caps.naming_convention
caps.escape_identifier = str
caps.escape_literal = serialize_value
caps.casefold_identifier = str
caps.has_case_sensitive_identifiers = True
caps.decimal_precision = (DEFAULT_NUMERIC_PRECISION, DEFAULT_NUMERIC_SCALE)
caps.wei_precision = (EVM_DECIMAL_PRECISION, 0)
caps.max_identifier_length = 65536
Expand Down
90 changes: 10 additions & 80 deletions dlt/common/destination/reference.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,19 +25,15 @@
import inspect

from dlt.common import logger
from dlt.common.destination.utils import verify_schema_capabilities
from dlt.common.exceptions import (
IdentifierTooLongException,
InvalidDestinationReference,
UnknownDestinationModule,
)
from dlt.common.schema import Schema, TTableSchema, TSchemaTables
from dlt.common.schema.exceptions import SchemaException
from dlt.common.schema.utils import (
get_write_disposition,
get_table_format,
get_columns_names_with_prop,
has_column_with_prop,
get_first_column_name_with_prop,
)
from dlt.common.configuration import configspec, resolve_configuration, known_sections
from dlt.common.configuration.specs import BaseConfiguration, CredentialsConfiguration
Expand Down Expand Up @@ -334,79 +330,13 @@ def __exit__(
pass

def _verify_schema(self) -> None:
"""Verifies and cleans up a schema before loading

* Checks all table and column name lengths against destination capabilities and raises on too long identifiers
* Removes and warns on (unbound) incomplete columns
"""

for table in self.schema.data_tables():
table_name = table["name"]
if len(table_name) > self.capabilities.max_identifier_length:
raise IdentifierTooLongException(
self.config.destination_type,
"table",
table_name,
self.capabilities.max_identifier_length,
)
if has_column_with_prop(table, "hard_delete"):
if len(get_columns_names_with_prop(table, "hard_delete")) > 1:
raise SchemaException(
f'Found multiple "hard_delete" column hints for table "{table_name}" in'
f' schema "{self.schema.name}" while only one is allowed:'
f' {", ".join(get_columns_names_with_prop(table, "hard_delete"))}.'
)
if table.get("write_disposition") in ("replace", "append"):
logger.warning(
f"""The "hard_delete" column hint for column "{get_first_column_name_with_prop(table, 'hard_delete')}" """
f'in table "{table_name}" with write disposition'
f' "{table.get("write_disposition")}"'
f' in schema "{self.schema.name}" will be ignored.'
' The "hard_delete" column hint is only applied when using'
' the "merge" write disposition.'
)
if has_column_with_prop(table, "dedup_sort"):
if len(get_columns_names_with_prop(table, "dedup_sort")) > 1:
raise SchemaException(
f'Found multiple "dedup_sort" column hints for table "{table_name}" in'
f' schema "{self.schema.name}" while only one is allowed:'
f' {", ".join(get_columns_names_with_prop(table, "dedup_sort"))}.'
)
if table.get("write_disposition") in ("replace", "append"):
logger.warning(
f"""The "dedup_sort" column hint for column "{get_first_column_name_with_prop(table, 'dedup_sort')}" """
f'in table "{table_name}" with write disposition'
f' "{table.get("write_disposition")}"'
f' in schema "{self.schema.name}" will be ignored.'
' The "dedup_sort" column hint is only applied when using'
' the "merge" write disposition.'
)
if table.get("write_disposition") == "merge" and not has_column_with_prop(
table, "primary_key"
):
logger.warning(
f"""The "dedup_sort" column hint for column "{get_first_column_name_with_prop(table, 'dedup_sort')}" """
f'in table "{table_name}" with write disposition'
f' "{table.get("write_disposition")}"'
f' in schema "{self.schema.name}" will be ignored.'
' The "dedup_sort" column hint is only applied when a'
" primary key has been specified."
)
for column_name, column in dict(table["columns"]).items():
if len(column_name) > self.capabilities.max_column_identifier_length:
raise IdentifierTooLongException(
self.config.destination_type,
"column",
f"{table_name}.{column_name}",
self.capabilities.max_column_identifier_length,
)
if not is_complete_column(column):
logger.warning(
f"A column {column_name} in table {table_name} in schema"
f" {self.schema.name} is incomplete. It was not bound to the data during"
" normalizations stage and its data type is unknown. Did you add this"
" column manually in code ie. as a merge key?"
)
"""Verifies schema before loading"""
if exceptions := verify_schema_capabilities(
self.schema, self.capabilities, self.config.destination_type, warnings=False
):
for exception in exceptions:
logger.error(str(exception))
raise exceptions[0]

def prepare_load_table(
self, table_name: str, prepare_for_staging: bool = False
Expand All @@ -421,7 +351,7 @@ def prepare_load_table(
table["table_format"] = get_table_format(self.schema.tables, table_name)
return table
except KeyError:
raise UnknownTableException(table_name)
raise UnknownTableException(self.schema.name, table_name)


class WithStateSync(ABC):
Expand Down Expand Up @@ -539,7 +469,7 @@ def to_name(ref: TDestinationReferenceArg) -> str:

@staticmethod
def normalize_type(destination_type: str) -> str:
"""Normalizes destination type string into a canonical form. Assumes that type names without dots correspond to build in destinations."""
"""Normalizes destination type string into a canonical form. Assumes that type names without dots correspond to built in destinations."""
if "." not in destination_type:
destination_type = "dlt.destinations." + destination_type
# the next two lines shorten the dlt internal destination paths to dlt.destinations.<destination_type>
Expand Down
105 changes: 105 additions & 0 deletions dlt/common/destination/utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
from typing import List

from dlt.common import logger
from dlt.common.exceptions import IdentifierTooLongException
from dlt.common.schema import Schema
from dlt.common.schema.exceptions import SchemaCorruptedException
from dlt.common.schema.typing import TTableSchema
from dlt.common.schema.utils import (
is_complete_column,
)
from dlt.common.typing import DictStrStr

from .capabilities import DestinationCapabilitiesContext


def verify_schema_capabilities(
schema: Schema,
capabilities: DestinationCapabilitiesContext,
destination_type: str,
warnings: bool = True,
) -> List[Exception]:
"""Verifies schema tables before loading against capabilities. Returns a list of exceptions representing critical problems with the schema.
It will log warnings by default. It is up to the caller to eventually raise exception

* Checks all table and column name lengths against destination capabilities and raises on too long identifiers
* Checks if schema has clashes due to case sensitivity of the identifiers
"""

log = logger.warning if warnings else logger.info
# collect all exceptions to show all problems in the schema
exception_log: List[Exception] = []
# combined casing function
case_identifier = lambda ident: capabilities.casefold_identifier(
(str if capabilities.has_case_sensitive_identifiers else str.casefold)(ident) # type: ignore
)
table_name_lookup: DictStrStr = {}
# name clash explanation
clash_msg = "Destination is case " + (
"sensitive" if capabilities.has_case_sensitive_identifiers else "insensitive"
)
if capabilities.casefold_identifier is not str:
clash_msg += (
f" but it uses {capabilities.casefold_identifier} to generate case insensitive"
" identifiers. You may try to change the destination capabilities by changing the"
" `casefold_identifier` to `str`"
)

# check for any table clashes
for table in schema.data_tables():
table_name = table["name"]
# detect table name conflict
cased_table_name = case_identifier(table_name)
if cased_table_name in table_name_lookup:
conflict_table_name = table_name_lookup[cased_table_name]
exception_log.append(
SchemaCorruptedException(
schema.name,
f"A table name {table_name} clashes with {conflict_table_name} after"
f" normalization to {cased_table_name}. "
+ clash_msg,
)
)
table_name_lookup[cased_table_name] = table_name
if len(table_name) > capabilities.max_identifier_length:
exception_log.append(
IdentifierTooLongException(
destination_type,
"table",
table_name,
capabilities.max_identifier_length,
)
)

column_name_lookup: DictStrStr = {}
for column_name, column in dict(table["columns"]).items():
# detect table name conflict
cased_column_name = case_identifier(column_name)
if cased_column_name in column_name_lookup:
conflict_column_name = column_name_lookup[cased_column_name]
exception_log.append(
SchemaCorruptedException(
schema.name,
f"A column name {column_name} in table {table_name} clashes with"
f" {conflict_column_name} after normalization to {cased_column_name}. "
+ clash_msg,
)
)
column_name_lookup[cased_column_name] = column_name
if len(column_name) > capabilities.max_column_identifier_length:
exception_log.append(
IdentifierTooLongException(
destination_type,
"column",
f"{table_name}.{column_name}",
capabilities.max_column_identifier_length,
)
)
if not is_complete_column(column):
log(
f"A column {column_name} in table {table_name} in schema"
f" {schema.name} is incomplete. It was not bound to the data during"
" normalizations stage and its data type is unknown. Did you add this"
" column manually in code ie. as a merge key?"
)
return exception_log
8 changes: 3 additions & 5 deletions dlt/common/normalizers/__init__.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,9 @@
from dlt.common.normalizers.configuration import NormalizersConfiguration
from dlt.common.normalizers.typing import TJSONNormalizer, TNormalizersConfig
from dlt.common.normalizers.utils import explicit_normalizers, import_normalizers
from dlt.common.normalizers.naming import NamingConvention


__all__ = [
"NormalizersConfiguration",
"NamingConvention",
"TJSONNormalizer",
"TNormalizersConfig",
"explicit_normalizers",
"import_normalizers",
]
7 changes: 4 additions & 3 deletions dlt/common/normalizers/configuration.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
import dataclasses
from typing import Optional, TYPE_CHECKING
from typing import Optional, TYPE_CHECKING, Union

from dlt.common.configuration import configspec
from dlt.common.configuration.specs import BaseConfiguration
from dlt.common.destination import DestinationCapabilitiesContext
from dlt.common.normalizers.naming import NamingConvention
from dlt.common.normalizers.typing import TJSONNormalizer
from dlt.common.typing import StrAny

Expand All @@ -13,8 +13,9 @@ class NormalizersConfiguration(BaseConfiguration):
# always in section
__section__: str = "schema"

naming: Optional[str] = None
naming: Optional[Union[str, NamingConvention]] = None
rudolfix marked this conversation as resolved.
Show resolved Hide resolved
json_normalizer: Optional[StrAny] = None
allow_identifier_change_on_table_with_data: Optional[bool] = None
destination_capabilities: Optional[DestinationCapabilitiesContext] = None # injectable

def on_resolved(self) -> None:
Expand Down
Loading