{
  "id": "sqlfluff__sqlfluff-2907",
  "question": "sqlfluff doesn't recognise a jinja variable set inside of \"if\" statement\n### Search before asking\n\n- [X] I searched the [issues](https://github.com/sqlfluff/sqlfluff/issues) and found no similar issues.\n\n\n### What Happened\n\nWhen I try to define a jinja variable using \"set\" jinja directive inside of an \"if\" jinja statement, sqlfluff complains: \r\n\"Undefined jinja template variable\".\n\n### Expected Behaviour\n\nto not have a linting issue\n\n### Observed Behaviour\n\nsqlfluff lint gives an error:\r\n\"Undefined jinja template variable\"\n\n### How to reproduce\n\ntry to create a \"temp.sql\" file with the following content\r\n\r\n```\r\n{% if True %}\r\n    {% set some_var %}1{% endset %}\r\n    SELECT {{some_var}}\r\n{% endif %}\r\n```\r\n\r\nand run:\r\n```\r\nsqlfluff lint ./temp.sql\r\n```\r\n\r\nYou will get the following error:\r\n```\r\n== [./temp.sql] FAIL                                                                                                                    \r\nL:   2 | P:  12 |  TMP | Undefined jinja template variable: 'some_var'\r\nL:   3 | P:  14 |  TMP | Undefined jinja template variable: 'some_var'\r\n```\n\n### Dialect\n\ntested on 'snowflake' dialect\n\n### Version\n\nsqlfluff, version 0.11.1\r\nPython 3.8.12\n\n### Configuration\n\n[sqlfluff]\r\nverbose = 1\r\ndialect = snowflake\r\ntemplater = jinja\r\nexclude_rules = L027,L031,L032,L036,L044,L046,L034,L050\r\noutput_line_length = 121\r\nsql_file_exts=.sql\r\n\r\n[sqlfluff:rules]\r\ntab_space_size = 4\r\nmax_line_length = 250\r\nindent_unit = space\r\ncomma_style = trailing\r\nallow_scalar = True\r\nsingle_table_references = consistent\r\nunquoted_identifiers_policy = aliases\r\n\r\n[sqlfluff:rules:L042]\r\nforbid_subquery_in = both\r\n\r\n[sqlfluff:rules:L010]  # Keywords\r\ncapitalisation_policy = upper\r\n\r\n[sqlfluff:rules:L014]\r\nextended_capitalisation_policy = lower\r\n\r\n[sqlfluff:rules:L030]  # function names\r\nextended_capitalisation_policy = upper\n\n### Are you willing to work on and submit a PR to address the issue?\n\n- [ ] Yes I am willing to submit a PR!\n\n### Code of Conduct\n\n- [X] I agree to follow this project's [Code of Conduct](https://github.com/sqlfluff/sqlfluff/blob/main/CODE_OF_CONDUCT.md)\n\n",
  "answer": "{\"task_id\": \"a5125d85-c4ba-4a92-83ae-15658c794f7a\", \"workflow_run_id\": \"9fbbd42a-e59f-4021-b98b-bf9df80de730\", \"data\": {\"id\": \"9fbbd42a-e59f-4021-b98b-bf9df80de730\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/jinja.py b/src/sqlfluff/core/templaters/jinja.py\\\\nindex abcdef1..1234567 100644\\\\n--- a/src/sqlfluff/core/templaters/jinja.py\\\\n+++ b/src/sqlfluff/core/templaters/jinja.py\\\\n@@ -150,7 +150,10 @@ class JinjaTracer:\\\\n     def visit_If(self, node):\\\\n         # We intentionally introduce a new scope for if blocks\\\\n         self.context_stack.append({})\\\\n-        self.generic_visit(node)\\\\n+        for stmt in node.body:\\\\n+            self.visit(stmt)\\\\n+        # Variables set in the 'if' block should not be popped from context\\\\n+        scoped_vars = self.context_stack.pop()\\\\n         self.context_stack[-1].update(scoped_vars)\\\\n \\\\n     def visit_Set(self, node):\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly addresses the issue by ensuring that variables set within an 'if' block are retained in the context stack. It modifies the visit_If method to handle the scope correctly, which aligns with the expected behavior of Jinja templating.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/jinja.py b/src/sqlfluff/core/templaters/jinja.py\\\\nindex abcdef3..1234569 100644\\\\n--- a/src/sqlfluff/core/templaters/jinja.py\\\\n+++ b/src/sqlfluff/core/templaters/jinja.py\\\\n@@ -190,8 +190,11 @@ class JinjaTracer:\\\\n     def visit_Name(self, node):\\\\n         # We want to track variable references\\\\n         for scope in reversed(self.context_stack):\\\\n-            if node.id in scope:\\\\n+            if node.id in scope and node.id not in self.unscopables:\\\\n                 return scope[node.id]\\\\n+        # If not found, look up in other parts of templates\\\\n+        if node.id in self.root_context:\\\\n+            return self.root_context[node.id]\\\\n         raise UndefinedJinjaVariable(\\\\\\\"Undefined jinja template variable: '{}'\\\\\\\", node.id)\\\\n \\\\n class JinjaTemplater:\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\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 enhances the variable lookup mechanism by checking a broader context if the variable is not found in the local scope. It provides a fallback mechanism which is useful but might not directly address the specific issue of scoping within 'if' blocks.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/jinja.py b/src/sqlfluff/core/templaters/jinja.py\\\\nindex abcdef2..1234568 100644\\\\n--- a/src/sqlfluff/core/templaters/jinja.py\\\\n+++ b/src/sqlfluff/core/templaters/jinja.py\\\\n@@ -170,7 +170,9 @@ class JinjaTracer:\\\\n     def visit_Set(self, node):\\\\n         # This will add the variable to the top of the context stack\\\\n         self.context_stack[-1][node.target.name] = node.value\\\\n-        self.generic_visit(node)\\\\n+        self.visit(node.value)\\\\n+        # Explicitly set in the current scope\\\\n+        self.context_stack[-1][node.target.name] = node.value.value\\\\n \\\\n     def visit_Name(self, node):\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.7,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the visit_Set method to handle variable values more explicitly. However, it introduces complexity and potential redundancy by setting the variable twice, which could lead to maintenance challenges.\\\",\\n      \\\"recommendation\\\": \\\"Use with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/jinja.py b/src/sqlfluff/core/templaters/jinja.py\\\\nindex abcdef4..1234570 100644\\\\n--- a/src/sqlfluff/core/templaters/jinja.py\\\\n+++ b/src/sqlfluff/core/templaters/jinja.py\\\\n@@ -100,6 +100,7 @@ class JinjaTemplater:\\\\n     def process(self, in_str):\\\\n         # Initialize the tracer with an empty context stack\\\\n         tracer = JinjaTracer()\\\\n+        tracer.root_context = {}\\\\n         tracer.process(in_str)\\\\n \\\\n     def render_string(self, template_string, context=None):\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.65,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.6,\\n        \\\"code_quality\\\": 0.65,\\n        \\\"risk_level\\\": 0.25,\\n        \\\"best_practices\\\": 0.7,\\n        \\\"test_coverage\\\": 0.6\\n      },\\n      \\\"reasoning\\\": \\\"This patch initializes a root context for broader context management, but it does not directly address the specific issue of variable scoping within 'if' blocks. It might be useful for other context-related issues but is not a targeted fix for the problem at hand.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended for this specific issue\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective solution for the issue of variable scoping within 'if' blocks in Jinja templates. It correctly manages the context stack to ensure variables are retained. Patch 3 provides a useful enhancement for broader context lookup but is not a direct fix. Patch 2 introduces complexity without clear benefits, and Patch 4 addresses a different aspect of context management.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    with open('temp.sql', 'w') as file:\\\\n        file.write('{% if True %}\\\\\\\\n    {% set some_var %}1{% endset %}\\\\\\\\nSELECT {{some_var}}\\\\\\\\n{% endif %}')\\\\n    lint_result = sqlfluff('lint temp.sql')\\\\n    assert lint_result.contains('Undefined jinja template variable: some_var')\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue of sqlfluff not recognizing a jinja variable set inside an 'if' statement\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail with 'Undefined jinja template variable: some_var'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    testcase_templates = [\\\\n        '{% if False %}\\\\\\\\n    {% set some_var %}1{% endset %}\\\\\\\\nSELECT {{some_var}}\\\\\\\\n{% endif %}',\\\\n        '{% if True %}\\\\\\\\n    {% set some_var %}1{% endset %}\\\\\\\\n{% if False %}\\\\\\\\n    SELECT {{some_var}}\\\\\\\\n{% endif %}\\\\\\\\n{% endif %}'\\\\n    ]\\\\n    for idx, template in enumerate(testcase_templates, start=1):\\\\n        with open(f'temp{idx}.sql', 'w') as file:\\\\n            file.write(template)\\\\n        lint_result = sqlfluff(f'lint temp{idx}.sql')\\\\n        assert lint_result.contains('Undefined jinja template variable: some_var')\\\\n\\\",\\n      \\\"description\\\": \\\"This test covers edge cases where the jinja variable set inside 'if' statements is affected by nested conditions\\\",\\n      \\\"expected_behavior\\\": \\\"Both tests should fail with 'Undefined jinja template variable: some_var'\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    with open('temp.sql', 'w') as file:\\\\n        file.write('{% if True %}\\\\\\\\n    {% set some_var %}1{% endset %}\\\\\\\\nSELECT {{some_var}}\\\\\\\\n{% endif %}')\\\\n    lint_result = sqlfluff('lint temp.sql')\\\\n    assert not lint_result.contains('Undefined jinja template variable: some_var')\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates that the patches successfully resolve the issue of jinja variables set inside 'if' statements\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass without any 'Undefined jinja template variable' errors\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated test cases to reproduce the original issue with jinja variables in 'if' statements, cover edge cases related to the problem, and validate the effectiveness of the provided patches.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 405.522041, \"total_tokens\": 28220, \"total_steps\": 9, \"created_at\": 1753336635, \"finished_at\": 1753337041}}"
}