{
  "id": "sqlfluff__sqlfluff-3066",
  "question": "Jinja: sqlfluff fails in the presence of assignments with multiple targets\n### Search before asking\r\n\r\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\r\n\r\nI did search, and I think this _may_ be related, but since no more information was provided I cannot confirm it: https://github.com/sqlfluff/sqlfluff/issues/2947. For this reason, I opened a new issue.\r\n\r\n### What Happened\r\n\r\nJinja templates support multiple targets in [assignments](https://jinja.palletsprojects.com/en/3.0.x/templates/#assignments). However, `sqlfluff` fails to lint a file in the presence of an assignment with multiple targets.\r\n\r\nI traced this back to the `update_inside_set_or_macro` function, specifically [this line](https://github.com/sqlfluff/sqlfluff/blob/main/src/sqlfluff/core/templaters/slicers/tracer.py#L244=).\r\n\r\nThe way `sqlfluff` is determining whether we are inside a [block assignment](https://jinja.palletsprojects.com/en/3.0.x/templates/#block-assignments) is by checking for the presence of an equals in the second index of the trimmed parts of the current raw slice:\r\n\r\n```python\r\nif len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts[2] != \"=\":\r\n```\r\n\r\nThis condition is false for single target assignments:\r\n\r\n```sql\r\n{% set a = 1 %}\r\n```\r\n\r\nWhich produce the expected trimmed parts (with spaces removed as in [line 243](https://github.com/sqlfluff/sqlfluff/blob/main/src/sqlfluff/core/templaters/slicers/tracer.py#L243=)):\r\n\r\n```python\r\n['set', 'a', '=', '1']\r\n#             2    \r\n```\r\n\r\nHowever, with multiple targets:\r\n\r\n```sql\r\n{% set a, b = 1, 2 %}\r\n```\r\n\r\n```python\r\n['set', 'a', ',', 'b', '=', '1', '2']\r\n#                       4    \r\n```\r\n\r\nEquals is no longer in the index 2, but has been bumped to index 4, yet we are not in the expanded block form of set assignments. This causes the `inside_set_or_macro` flag to be incorrectly set to `True`, as if we were using a block assignment, which causes the entire template to be ignored (or something like that), and leads to the eventual `ValueError` raised.\r\n\r\nI played around a bit with potential solutions: first, I tried incrementing the index of the equals by the number of commas:\r\n\r\n```python\r\nequals_index = 2 + sum((c == ',' for c in  filtered_trimmed_parts))\r\nif len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts[equals_index] != \"=\":\r\n```\r\n\r\nHowever, this would bring issues if using the expanded form of set assignments with any commas in it, or in the presence of an uneven number of commas on both sides of the assignment.\r\n\r\nAnother simpler option would be to check for the presence of a single equals:\r\n\r\n```python\r\nif len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts.count(\"=\") != 1:\r\n```\r\n\r\nThis one seems more promising, specially considering that multiple targets appear not to be supported with block assignments (at least, that's what I think, as the docs don't mention it, and trying it locally raises a too many values to unpack error). Thus, the first condition will always be true for block assignments (so, even the presence of an equals in the body of the assignment would not cause issues).\r\n\r\n### Expected Behaviour\r\n\r\nsqlfluff should lint files properly, even in the presence of assignments with multiple targets.\r\n\r\n### Observed Behaviour\r\n\r\nLinting fails when an exception is raised:\r\n\r\n```\r\nTraceback (most recent call last):\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10dbt/bin/sqlfluff\", line 8, in <module>\r\n    sys.exit(cli())\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/click/core.py\", line 1128, in __call__\r\n    return self.main(*args, **kwargs)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/click/core.py\", line 1053, in main\r\n    rv = self.invoke(ctx)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/click/core.py\", line 1659, in invoke\r\n    return _process_result(sub_ctx.command.invoke(sub_ctx))\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/click/core.py\", line 1395, in invoke\r\n    return ctx.invoke(self.callback, **ctx.params)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/click/core.py\", line 754, in invoke\r\n    return __callback(*args, **kwargs)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/cli/commands.py\", line 541, in lint\r\n    result = lnt.lint_paths(\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/linter.py\", line 1098, in lint_paths\r\n    self.lint_path(\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/linter.py\", line 1050, in lint_path\r\n    for i, linted_file in enumerate(runner.run(fnames, fix), start=1):\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/runner.py\", line 101, in run\r\n    for fname, partial in self.iter_partials(fnames, fix=fix):\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/runner.py\", line 54, in iter_partials\r\n    for fname, rendered in self.iter_rendered(fnames):\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/runner.py\", line 43, in iter_rendered\r\n    yield fname, self.linter.render_file(fname, self.config)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/linter.py\", line 771, in render_file\r\n    return self.render_string(raw_file, fname, config, encoding)\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/linter/linter.py\", line 742, in render_string\r\n    templated_file, templater_violations = self.templater.process(\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/templaters/jinja.py\", line 394, in process\r\n    TemplatedFile(\r\n  File \"/home/tomasfarias/.pyenv/versions/3.10.2/envs/3.10dbt/lib/python3.10/site-packages/sqlfluff/core/templaters/base.py\", line 94, in __init__\r\n    raise ValueError(\"Cannot instantiate a templated file unsliced!\")\r\nValueError: Cannot instantiate a templated file unsliced!\r\n```\r\n\r\n### How to reproduce\r\n\r\n1. Save the following template to `model.sql` in an empty directory:\r\n```sql\r\n{% set a, b = 1, 2 %}\r\n\r\nSELECT {{ a }}\r\n```\r\n2. Run `sqlfluff lint model.sql --dialect 'postgres'`\r\n\r\n\r\n### Dialect\r\n\r\nTried with postgres and redshift dialects, however I think others may be affected as long as they use jinja templates.\r\n\r\n### Version\r\n\r\nv0.12.0\r\n\r\n### Configuration\r\n\r\nNothing, ran from an empty directory.\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [X] Yes I am willing to submit a PR!\r\n\r\n### Code of Conduct\r\n\r\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\r\n\n",
  "answer": "{\"task_id\": \"e5226760-9189-4335-88d7-ab77fbe01e6c\", \"workflow_run_id\": \"da104df1-8797-42b5-b0bf-fc46c6a9bbea\", \"data\": {\"id\": \"da104df1-8797-42b5-b0bf-fc46c6a9bbea\", \"workflow_id\": \"4faf996e-a613-49ba-90e4-a0af9c740cdf\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/slicers/tracer.py b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\nindex 1234567..fedcba9 100644\\\\n--- a/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n+++ b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n@@ -244,7 +244,7 @@ class JinjaTracer:\\\\n     def update_inside_set_or_macro(self, raw_slice):\\\\n         # Determine whether we're inside a set or macro block\\\\n         filtered_trimmed_parts = raw_slice.get_raw().strip().split()\\\\n-        if len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts[2] != \\\\\\\"=\\\\\\\":\\\\n+        if len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts.count(\\\\\\\"=\\\\\\\") != 1:\\\\n             self.inside_set_or_macro = False\\\\n         else:\\\\n             self.inside_set_or_macro = True\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly identifies the presence of a single '=' in the assignment, which is a straightforward and effective solution. It minimizes the risk of false positives and aligns well with the expected behavior of Jinja templates.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/slicers/tracer.py b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\nindex 1234567..7654321 100644\\\\n--- a/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n+++ b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n@@ -244,7 +244,10 @@ class JinjaTracer:\\\\n     def update_inside_set_or_macro(self, raw_slice):\\\\n         # Determine whether we're inside a set or macro block\\\\n         filtered_trimmed_parts = raw_slice.get_raw().strip().split()\\\\n-        if len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts[2] != \\\\\\\"=\\\\\\\":\\\\n+        equals_index = next((i for i, part in enumerate(filtered_trimmed_parts) if part == '='), None)\\\\n+        if equals_index is None or equals_index < 2:\\\\n             self.inside_set_or_macro = False\\\\n         else:\\\\n             self.inside_set_or_macro = True\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch dynamically finds the first occurrence of '=' and checks its position. It is a robust solution but slightly more complex than necessary for the problem at hand.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/slicers/tracer.py b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n+++ b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n@@ -244,7 +244,7 @@ class JinjaTracer:\\\\n     def update_inside_set_or_macro(self, raw_slice):\\\\n         # Determine whether we're inside a set or macro block\\\\n         filtered_trimmed_parts = raw_slice.get_raw().strip().split()\\\\n-        if len(filtered_trimmed_parts) < 3 or filtered_trimmed_parts[2] != \\\\\\\"=\\\\\\\":\\\\n+        if len(filtered_trimmed_parts) < 3 or \\\\\\\"=\\\\\\\" not in filtered_trimmed_parts[2:]:\\\\n             self.inside_set_or_macro = False\\\\n         else:\\\\n             self.inside_set_or_macro = True\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch checks for the presence of '=' in the parts after the initial keywords. While it addresses the issue, it may not handle all edge cases as cleanly as the other patches.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the most straightforward and effective solution, ensuring exactly one '=' is present, which aligns with the expected behavior of Jinja templates. Patch 3 is robust but slightly more complex, while Patch 1, although functional, may not handle all edge cases as effectively.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    template_content = \\\\\\\"{% set a, b = 1, 2 %}\\\\\\\\nSELECT {{ a }}\\\\\\\"\\\\n    with open('model.sql', 'w') as file:\\\\n        file.write(template_content)\\\\n    result = sqlfluff_lint('model.sql', dialect='postgres')\\\\n    assert 'ValueError: Cannot instantiate a templated file unsliced!' in result\\\\n    \\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by creating a template with an assignment containing multiple targets and running sqlfluff lint.\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to fail with a ValueError due to the issue in handling assignments with multiple targets.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases_assignments\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases_assignments():\\\\n    template_content_single = \\\\\\\"{% set a = 1 %}\\\\\\\"\\\\n    with open('single_target_assignment.sql', 'w') as file1:\\\\n        file1.write(template_content_single)\\\\n    result_single = sqlfluff_lint('single_target_assignment.sql', dialect='postgres')\\\\n    assert 'ValueError' not in result_single\\\\n    \\\\n    template_content_expanded = \\\\\\\"{% set a, b, c = 1, 2, 3 %}\\\\\\\"\\\\n    with open('expanded_multi_target_assignment.sql', 'w') as file2:\\\\n        file2.write(template_content_expanded)\\\\n    result_expanded = sqlfluff_lint('expanded_multi_target_assignment.sql', dialect='postgres')\\\\n    assert 'ValueError' not in result_expanded\\\\n    \\\",\\n      \\\"description\\\": \\\"This test covers edge cases related to assignments with single target and multiple targets.\\\",\\n      \\\"expected_behavior\\\": \\\"Both tests are expected to pass without raising a ValueError.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_1_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_1_validation():\\\\n    # Apply Patch 1\\\\n    # Run similar test_reproduce_original_issue test case\\\\n    template_content = \\\\\\\"{% set a, b = 1, 2 %}\\\\\\\\nSELECT {{ a }}\\\\\\\"\\\\n    with open('model.sql', 'w') as file:\\\\n        file.write(template_content)\\\\n    result = sqlfluff_lint('model.sql', dialect='postgres')\\\\n    assert 'ValueError' not in result\\\\n    \\\",\\n      \\\"description\\\": \\\"This test validates Patch 1 which dynamically checks for the presence of equals in Jinja assignments with multiple targets.\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to pass without raising a ValueError after applying Patch 1.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_2_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_2_validation():\\\\n    # Apply Patch 2\\\\n    # Run similar test_reproduce_original_issue test case\\\\n    template_content = \\\\\\\"{% set a, b = 1, 2 %}\\\\\\\\nSELECT {{ a }}\\\\\\\"\\\\n    with open('model.sql', 'w') as file:\\\\n        file.write(template_content)\\\\n    # Run sqlfluff lint with the patched version\\\\n    result = sqlfluff_lint('model.sql', dialect='postgres')\\\\n    assert 'ValueError' not in result\\\\n    \\\",\\n      \\\"description\\\": \\\"This test validates Patch 2 which ensures there is exactly one equals in Jinja assignments with multiple targets.\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to pass without raising a ValueError after applying Patch 2.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_3_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_3_validation():\\\\n    # Apply Patch 3\\\\n    # Run similar test_reproduce_original_issue test case\\\\n    template_content = \\\\\\\"{% set a, b = 1, 2 %}\\\\\\\\nSELECT {{ a }}\\\\\\\"\\\\n    with open('model.sql', 'w') as file:\\\\n        file.write(template_content)\\\\n    # Run sqlfluff lint with the patched version\\\\n    result = sqlfluff_lint('model.sql', dialect='postgres')\\\\n    assert 'ValueError' not in result\\\\n    \\\",\\n      \\\"description\\\": \\\"This test validates Patch 3 which finds the first occurrence of equals dynamically while handling Jinja assignments with multiple targets.\\\",\\n      \\\"expected_behavior\\\": \\\"The test is expected to pass without raising a ValueError after applying Patch 3.\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated test cases to reproduce the original issue with Jinja assignments having multiple targets in SQLFluff, cover edge cases related to assignments, and validate the proposed patches for the issue.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 260.173963, \"total_tokens\": 29266, \"total_steps\": 9, \"created_at\": 1753358661, \"finished_at\": 1753358921}}"
}