dbx doesn't like Jinja Macros
aviadshimoni opened this issue ยท 4 comments
Expected Behavior
There is a lot of boilerplate code in the deployment.yaml that we (DevOps) want to mask from the developers and put it in a common repo (DRY).
for that, I've created a common_function.yaml file and put jinja functions there.
Current Behavior
Steps to Reproduce (for bugs)
Create common_function.yaml, pass with to dbx as --jinja-variables-file and reference functions from deployment.yaml.
Context
If I've a different solution to implement my goal, I would love to hear that.
Your Environment
- dbx version used: 0.8.17
- Databricks Runtime version: 12.2
@renardeinside hi Ivan, now I see that I used the variables file incorrectly since it doesn't support Jinja syntax.
https://dbx.readthedocs.io/en/latest/features/jinja_support/#custom-functions-support
gcp attributes works, but the get_cluster common doesn't, so as I see it I need to call functions after the ':' and add the whole value for the property.
any idea when this feature will be stable and not experimental? we want to build the DRY solution based on this but afraid it would break.
dbx logs from example above:
[dbx][2023-06-26 09:59:20.385] ๐ Found custom Jinja functions defined in .dbx/_custom_jinja_functions.py, loading them
[dbx][2023-06-26 09:59:20.386] โ
Custom Jinja functions successfully loaded
โญโโโโโโโโโโโโโโโโโโโโโ Traceback (most recent call last) โโโโโโโโโโโโโโโโโโโโโโโฎ
โ /usr/local/lib/python3.10/site-packages/dbx/commands/deploy.py:111 in deploy โ
โ โ
โ 108 โ if not branch_name: โ
โ 109 โ โ branch_name = get_current_branch_name() โ
โ 110 โ โ
โ โฑ 111 โ config_reader = ConfigReader(deployment_file, jinja_variables_file โ
โ 112 โ config = config_reader.with_build_properties( โ
โ 113 โ โ BuildProperties(potential_build=True, no_rebuild=no_rebuild) โ
โ 114 โ ).get_config() โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:122 in โ
โ __init__ โ
โ โ
โ 119 โ def __init__(self, path: Path, jinja_vars_file: Optional[Path] = N โ
โ 120 โ โ self._jinja_vars_file = jinja_vars_file โ
โ 121 โ โ self._path = path โ
โ โฑ 122 โ โ self._reader = self._define_reader() โ
โ 123 โ โ self._build_properties = BuildProperties() โ
โ 124 โ โ
โ 125 โ def with_build_properties(self, build_properties: BuildProperties) โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:140 in โ
โ _define_reader โ
โ โ
โ 137 โ โ โ โ ) โ
โ 138 โ โ โ โ return Jinja2ConfigReader(self._path, ext=self._path.s โ
โ 139 โ โ elif ProjectConfigurationManager().get_jinja_support(): โ
โ โฑ 140 โ โ โ return Jinja2ConfigReader(self._path, ext=self._path.suffi โ
โ 141 โ โ else: โ
โ 142 โ โ โ if self._jinja_vars_file: โ
โ 143 โ โ โ โ raise Exception( โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:64 in โ
โ __init__ โ
โ โ
โ 61 โ def __init__(self, path: Path, ext: str, jinja_vars_file: Optional โ
โ 62 โ โ self._ext = ext โ
โ 63 โ โ self._jinja_vars_file = jinja_vars_file โ
โ โฑ 64 โ โ super().__init__(path) โ
โ 65 โ โ
โ 66 โ @staticmethod โ
โ 67 โ def _read_vars_file(file_path: Path) -> Dict[str, Any]: โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:23 in โ
โ __init__ โ
โ โ
โ 20 class _AbstractConfigReader(ABC): โ
โ 21 โ def __init__(self, path: Path): โ
โ 22 โ โ self._path = path โ
โ โฑ 23 โ โ self.config = self.get_config() โ
โ 24 โ โ
โ 25 โ def get_config(self) -> DeploymentConfig: โ
โ 26 โ โ return self._read_file() โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:26 in โ
โ get_config โ
โ โ
โ 23 โ โ self.config = self.get_config() โ
โ 24 โ โ
โ 25 โ def get_config(self) -> DeploymentConfig: โ
โ โฑ 26 โ โ return self._read_file() โ
โ 27 โ โ
โ 28 โ @abstractmethod โ
โ 29 โ def _read_file(self) -> DeploymentConfig: โ
โ โ
โ /usr/local/lib/python3.10/site-packages/dbx/api/config_reader.py:101 in โ
โ _read_file โ
โ โ
โ 98 โ โ โ _content = json.loads(rendered) โ
โ 99 โ โ โ return JsonConfigReader.read_content(_content) โ
โ 100 โ โ elif self._ext in [".yml", ".yaml"]: โ
โ โฑ 101 โ โ โ _content = yaml.load(rendered, yaml.SafeLoader) โ
โ 102 โ โ โ return DeploymentConfig.from_payload(_content) โ
โ 103 โ โ else: โ
โ 104 โ โ โ raise Exception(f"Unexpected extension for Jinja reader: { โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/__init__.py:81 in load โ
โ โ
โ 78 โ """ โ
โ 79 โ loader = Loader(stream) โ
โ 80 โ try: โ
โ โฑ 81 โ โ return loader.get_single_data() โ
โ 82 โ finally: โ
โ 83 โ โ loader.dispose() โ
โ 84 โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/constructor.py:49 in โ
โ get_single_data โ
โ โ
โ 46 โ โ
โ 47 โ def get_single_data(self): โ
โ 48 โ โ # Ensure that the stream contains a single document and constr โ
โ โฑ 49 โ โ node = self.get_single_node() โ
โ 50 โ โ if node is not None: โ
โ 51 โ โ โ return self.construct_document(node) โ
โ 52 โ โ return None โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:36 in โ
โ get_single_node โ
โ โ
โ 33 โ โ # Compose a document if the stream is not empty. โ
โ 34 โ โ document = None โ
โ 35 โ โ if not self.check_event(StreamEndEvent): โ
โ โฑ 36 โ โ โ document = self.compose_document() โ
โ 37 โ โ โ
โ 38 โ โ # Ensure that the stream contains no more documents. โ
โ 39 โ โ if not self.check_event(StreamEndEvent): โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:55 in โ
โ compose_document โ
โ โ
โ 52 โ โ self.get_event() โ
โ 53 โ โ โ
โ 54 โ โ # Compose the root node. โ
โ โฑ 55 โ โ node = self.compose_node(None, None) โ
โ 56 โ โ โ
โ 57 โ โ # Drop the DOCUMENT-END event. โ
โ 58 โ โ self.get_event() โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:84 in compose_node โ
โ โ
โ 81 โ โ elif self.check_event(SequenceStartEvent): โ
โ 82 โ โ โ node = self.compose_sequence_node(anchor) โ
โ 83 โ โ elif self.check_event(MappingStartEvent): โ
โ โฑ 84 โ โ โ node = self.compose_mapping_node(anchor) โ
โ 85 โ โ self.ascend_resolver() โ
โ 86 โ โ return node โ
โ 87 โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:133 in โ
โ compose_mapping_node โ
โ โ
โ 130 โ โ โ #if item_key in node.value: โ
โ 131 โ โ โ # raise ComposerError("while composing a mapping", star โ
โ 132 โ โ โ # "found duplicate key", key_event.start_mark) โ
โ โฑ 133 โ โ โ item_value = self.compose_node(node, item_key) โ
โ 134 โ โ โ #node.value[item_key] = item_value โ
โ 135 โ โ โ node.value.append((item_key, item_value)) โ
โ 136 โ โ end_event = self.get_event() โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:84 in compose_node โ
โ โ
โ 81 โ โ elif self.check_event(SequenceStartEvent): โ
โ 82 โ โ โ node = self.compose_sequence_node(anchor) โ
โ 83 โ โ elif self.check_event(MappingStartEvent): โ
โ โฑ 84 โ โ โ node = self.compose_mapping_node(anchor) โ
โ 85 โ โ self.ascend_resolver() โ
โ 86 โ โ return node โ
โ 87 โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/composer.py:127 in โ
โ compose_mapping_node โ
โ โ
โ 124 โ โ โ โ flow_style=start_event.flow_style) โ
โ 125 โ โ if anchor is not None: โ
โ 126 โ โ โ self.anchors[anchor] = node โ
โ โฑ 127 โ โ while not self.check_event(MappingEndEvent): โ
โ 128 โ โ โ #key_event = self.peek_event() โ
โ 129 โ โ โ item_key = self.compose_node(node, None) โ
โ 130 โ โ โ #if item_key in node.value: โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/parser.py:98 in check_event โ
โ โ
โ 95 โ โ # Check the type of the next event. โ
โ 96 โ โ if self.current_event is None: โ
โ 97 โ โ โ if self.state: โ
โ โฑ 98 โ โ โ โ self.current_event = self.state() โ
โ 99 โ โ if self.current_event is not None: โ
โ 100 โ โ โ if not choices: โ
โ 101 โ โ โ โ return True โ
โ โ
โ /usr/local/lib/python3.10/site-packages/yaml/parser.py:438 in โ
โ parse_block_mapping_key โ
โ โ
โ 435 โ โ โ โ return self.process_empty_scalar(token.end_mark) โ
โ 436 โ โ if not self.check_token(BlockEndToken): โ
โ 437 โ โ โ token = self.peek_token() โ
โ โฑ 438 โ โ โ raise ParserError("while parsing a block mapping", self.ma โ
โ 439 โ โ โ โ โ "expected <block end>, but found %r" % token.id, t โ
โ 440 โ โ token = self.get_token() โ
โ 441 โ โ event = MappingEndEvent(token.start_mark, token.end_mark) โ
โฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
ParserError: while parsing a block mapping
in "<unicode string>", line 11, column 3:
google_service_account_gcp_attri ...
^
expected <block end>, but found '<block mapping start>'
in "<unicode string>", line 26, column 5:
gcp_attributes: {'zone_id': 'aut ...
^
Cleaning up project directory and file based variables 00:01
ERROR: Job failed: command terminated with exit code 1
Another issue trying to implement https://dbx.readthedocs.io/en/latest/features/jinja_support/#support-for-includes:
this is my gcp_attributes_yaml.j2 file:
zone_id: auto use_preemptible_executors: true availability: PREEMPTIBLE_WITH_FALLBACK_GCP
this is how I reference it in my deployment.yaml:
job_clusters: - job_cluster_key: "{{env['SCHEMA']~'_'~env['TABLE_NAME']~'_etl'}}" new_cluster: gcp_attributes: { % include 'includes/gcp_attributes.yaml.j2' % }
trying moving the include one line down
getting this error:
ScannerError: mapping values are not allowed here
in "", line 26, column 32:
gcp_attributes: zone_id: auto
are you sure the include works for YAML files and not JSON only?
Another issue trying to implement https://dbx.readthedocs.io/en/latest/features/jinja_support/#support-for-includes:
this is my gcp_attributes_yaml.j2 file:
zone_id: auto use_preemptible_executors: true availability: PREEMPTIBLE_WITH_FALLBACK_GCP
this is how I reference it in my deployment.yaml:
job_clusters: - job_cluster_key: "{{env['SCHEMA']~'_'~env['TABLE_NAME']~'_etl'}}" new_cluster: gcp_attributes: { % include 'includes/gcp_attributes.yaml.j2' % }
trying moving the include one line down getting this error: ScannerError: mapping values are not allowed here in "", line 26, column 32: gcp_attributes: zone_id: autoare you sure the include works for YAML files and not JSON only?
On the same page, is it possible to include support for 'includes' in the yaml templates?