* checkout from alex/cac * checkout from alex/cac * checkout from alex/cac * checkout from alex/cac * Add missing tests * Add missing files * Add missing tests * add missing file * missing file * missing file * sengrid low code connector * rename * doc * doc * remove broken test * rename * jinja dependency * Add comment * comment * comment * pyjq dependency * update import * rename file * delete unused file * Revert "delete unused file" This reverts commit758e939367. * fix * rename * abstract property * delete unused field * delete unused field * rename * pass kwargs directly * isort * Revert "isort" This reverts commit4a79223944. * isort * update state * fix imports * update * update dependency * remove dead code * remove dead code * format * rename file * decoder * Use decoder * Update comment * dict_state is actually backed by a dict * Add a comment * update state takes kwargs * move state out of offset paginator * fix * update jq parameter order * fix * pass config * update * update * remove incremental mixin * delete comment * start workin on yaml parser * fix test * progress * refer and overwrite partials * factory tests pass * fix * reset * Assert http_method is an enum value * fix auth * read lists works * fix test * comment * implement all streams * build connection checker * update comments * update comments * remove no_state * rename package * checkout from alex/cac * Add missing tests * Add missing files * missing file * rename * jinja dependency * Add comment * comment * comment * Revert "delete unused file" This reverts commit758e939367. * delete unused field * delete unused field * rename * pass kwargs directly * isort * Revert "isort" This reverts commit4a79223944. * format * decoder * better error handling * remove nostate * isort * remove print * move test * delete duplicates * update * delete dead code * Update mapping type to [str, Any] * add comment * Add comment * pass parameters through kwargs * pass parameters through kwargs * fix test * update interface * update interface to pass source in interface * update interface to pass source in interface * rename to stream_slicer * Allow passing a string or an enum * Define StateType enum * unit tests pass * update dict state * update * can read * fix test * fix from yaml update * elif * convert state_type if not of type type * convert state_type if not of type type * Add a test * Low code connectors: string interpolation with jinja (#12852) * checkout from alex/cac * Add missing tests * Add missing files * missing file * rename * jinja dependency * Add comment * comment * comment * Revert "delete unused file" This reverts commit758e939367. * delete unused field * delete unused field * rename * pass kwargs directly * isort * Revert "isort" This reverts commit4a79223944. * format * decoder * better error handling * remove nostate * isort * delete dead code * Update mapping type to [str, Any] * add comment * Add comment * pass parameters through kwargs * move test to right module * Add missing test * Use authbase instead of deprecated class * leverage generator * remove sendgrid specific code * update * update * delete comment * remove sendgrid specific file * remove unused file * Delete dead code * rename methods * rename to declarative * rename the classes too * select streams to check * nit * rename method * rename class * {} is faster than dict() * Update airbyte-cdk/python/airbyte_cdk/sources/declarative/parsers/yaml_parser.py Co-authored-by: Sherif A. Nada <snadalive@gmail.com> * more precise exception * rename class * add comment * Try to install packages to build jq * isort * only automake * Revert "only automake" This reverts commitc8fe154ffc. * remove git * rename file * create components in kwargs * Use tuple of strings * parser doesn't need to be stored * move file and delete duplicates * Revert "Use tuple of strings" This reverts commitab5a7afd08. * raise error if streams to check are not in the catalog * Revert "Revert "Use tuple of strings"" This reverts commit7c9fb8eb33. * traverse tree * rename to options * move docstring * Update airbyte-cdk/python/airbyte_cdk/sources/declarative/checks/check_stream.py Co-authored-by: Sherif A. Nada <snadalive@gmail.com> * fix tests and format * format * update * better error message * Add jq dependency * Use request header provider * rename * rename field * remove get_context method * rename * add a comment * format Co-authored-by: Sherif A. Nada <snadalive@gmail.com>
70 lines
2.4 KiB
Python
70 lines
2.4 KiB
Python
#
|
|
# Copyright (c) 2022 Airbyte, Inc., all rights reserved.
|
|
#
|
|
|
|
from __future__ import annotations
|
|
|
|
import copy
|
|
import importlib
|
|
from typing import Any, Mapping
|
|
|
|
from airbyte_cdk.sources.declarative.create_partial import create
|
|
from airbyte_cdk.sources.declarative.interpolation.jinja import JinjaInterpolation
|
|
from airbyte_cdk.sources.declarative.types import Config
|
|
|
|
|
|
class DeclarativeComponentFactory:
|
|
def __init__(self):
|
|
self._interpolator = JinjaInterpolation()
|
|
|
|
def create_component(self, component_definition: Mapping[str, Any], config: Config):
|
|
"""
|
|
|
|
:param component_definition: mapping defining the object to create. It should have at least one field: `class_name`
|
|
:param config: Connector's config
|
|
:return: the object to create
|
|
"""
|
|
kwargs = copy.deepcopy(component_definition)
|
|
class_name = kwargs.pop("class_name")
|
|
return self.build(class_name, config, **kwargs)
|
|
|
|
def build(self, class_name: str, config, **kwargs):
|
|
fqcn = class_name
|
|
split = fqcn.split(".")
|
|
module = ".".join(split[:-1])
|
|
class_name = split[-1]
|
|
|
|
# create components in options before propagating them
|
|
if "options" in kwargs:
|
|
kwargs["options"] = {k: self._create_subcomponent(v, kwargs, config) for k, v in kwargs["options"].items()}
|
|
|
|
updated_kwargs = {k: self._create_subcomponent(v, kwargs, config) for k, v in kwargs.items()}
|
|
|
|
class_ = getattr(importlib.import_module(module), class_name)
|
|
return create(class_, config=config, **updated_kwargs)
|
|
|
|
def _merge_dicts(self, d1, d2):
|
|
return {**d1, **d2}
|
|
|
|
def _create_subcomponent(self, v, kwargs, config):
|
|
if isinstance(v, dict) and "class_name" in v:
|
|
# propagate kwargs to inner objects
|
|
v["options"] = self._merge_dicts(kwargs.get("options", dict()), v.get("options", dict()))
|
|
|
|
return self.create_component(v, config)()
|
|
elif isinstance(v, list):
|
|
return [
|
|
self._create_subcomponent(
|
|
sub, self._merge_dicts(kwargs.get("options", dict()), self._get_subcomponent_options(sub)), config
|
|
)
|
|
for sub in v
|
|
]
|
|
else:
|
|
return v
|
|
|
|
def _get_subcomponent_options(self, sub: Any):
|
|
if isinstance(sub, dict):
|
|
return sub.get("options", {})
|
|
else:
|
|
return {}
|