{
  "id": "sqlfluff__sqlfluff-3436",
  "question": "Fatal templating error with Jinja templater. Tracer produces odd results.\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\nIssue found while assessing an Airflow project.\r\n\r\nThe smallest query I can make which triggers the issue is: \r\n```sql\r\nSELECT\r\n\t{% block table_name %}a{% endblock %}.b\r\nFROM d.{{ self.table_name() }}\r\n```\r\n\r\nWhen running this query through `lint` I get an `AssertionError`, or if running on the more friendly error message PR (#3433) I get: `WARNING    Length of templated file mismatch with final slice: 21 != 19.`.\n\n### Expected Behaviour\n\nThis query should slice properly and probably eventually give a jinja error that the required variables are undefined.\n\n### Observed Behaviour\n\nI've dug a little into the error and the sliced file being produced is:\r\n\r\n```python\r\n[\r\n    TemplatedFileSlice(slice_type='literal', source_slice=slice(0, 8, None), templated_slice=slice(0, 8, None)),\r\n    TemplatedFileSlice(slice_type='block_start', source_slice=slice(8, 30, None), templated_slice=slice(8, 8, None)),\r\n    TemplatedFileSlice(slice_type='literal', source_slice=slice(30, 31, None), templated_slice=slice(8, 9, None)),\r\n    TemplatedFileSlice(slice_type='block_end', source_slice=slice(31, 45, None), templated_slice=slice(9, 9, None)),\r\n    TemplatedFileSlice(slice_type='literal', source_slice=slice(45, 55, None), templated_slice=slice(9, 19, None)),\r\n    TemplatedFileSlice(slice_type='templated', source_slice=slice(55, 78, None), templated_slice=slice(19, 19, None)),\r\n    TemplatedFileSlice(slice_type='literal', source_slice=slice(78, 79, None), templated_slice=slice(19, 19, None))\r\n]\r\n```\r\n\r\nThe issue is that while the `source_slice` looks correct for the slices, almost all of the `templated_slices` values have zero length, and importantly the last one doesn't end at position 21.\r\n\r\nThe rendered file is `SELECT\\n\\ta.b\\nFROM d.a\\n` (I've included the escape chars) which is indeed 21 chars long.\r\n\r\n@barrywhart I might need your help to work out what's going on with the Jinja tracer here.\n\n### How to reproduce\n\nRun provided query, `main` branch. Set to the `jinja` templater.\n\n### Dialect\n\ndialect is set to `snowflake`, but I don't think we're getting far enough for that to make a difference.\n\n### Version\n\n`main` branch commit `cb6357c540d2d968f766f3a7a4fa16f231cb80e4` (and a few branches derived from it)\n\n### Configuration\n\nN/A\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\": \"bfc2bc42-b918-4477-883a-fdcde47c2af4\", \"workflow_run_id\": \"489e4cd3-d36b-4091-835d-7ec29ae5c4ea\", \"data\": {\"id\": \"489e4cd3-d36b-4091-835d-7ec29ae5c4ea\", \"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/slicers/tracer.py b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\nindex 7b5f4c0..c9f5a1d 100644\\\\n--- a/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n+++ b/src/sqlfluff/core/templaters/slicers/tracer.py\\\\n@@ -155,7 +155,12 @@ class JinjaTracer:\\\\n                 templated_slice=TemplatedFileSlice(\\\\n                     slice_type=template_slice_type,\\\\n                     source_slice=full_template_slice,\\\\n-                    templated_slice=full_template_slice\\\\n+                    # Adjust templated_slice calculation\\\\n+                    templated_slice=(\\\\n+                        slice(templated_positions[0], templated_positions[1])\\\\n+                        if template_slice_type != 'block' else\\\\n+                        slice(templated_positions[0], templated_positions[0])\\\\n+                    )\\\\n                 )\\\\n             )\\\\n\\\",\\n      \\\"rank\\\": 1,\\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 directly addresses the issue by adjusting the calculation of templated_slice positions, which is the core of the problem. It introduces minimal risk and aligns well with coding best practices.\\\",\\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 4f1c2d8..cab34b3 100644\\\\n--- a/src/sqlfluff/core/templaters/jinja.py\\\\n+++ b/src/sqlfluff/core/templaters/jinja.py\\\\n@@ -110,7 +110,10 @@ class JinjaTemplater:\\\\n             linter_logger.warning(\\\\n                 'Length of templated file mismatch. Attempting to adjust slices.'\\\\n             )\\\\n-            # Adjust for mismatch in templated length if possible\\\\n+            # Adjust templated file length based on last correct slice\\\\n+            if tracer.slices and tracer.slices[-1].templated_slice.stop != raw_slices[-1].stop:\\\\n+                tracer.slices[-1].templated_slice = slice(\\\\n+                    tracer.slices[-1].templated_slice.start, raw_slices[-1].stop)\\\\n\\\\n         return TemplatedFile(\\\\n             source_str=raw_file,\\\\n\\\",\\n      \\\"rank\\\": 2,\\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 provides a fallback mechanism to adjust the templated file length, which is useful but more of a workaround than a direct fix. It is less precise than Patch 1 but still valuable.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/core/templaters/slicers/tracer_plugins.py b/src/sqlfluff/core/templaters/slicers/tracer_plugins.py\\\\nindex 1d4aa3f..a2f8d6c 100644\\\\n--- a/src/sqlfluff/core/templaters/slicers/tracer_plugins.py\\\\n+++ b/src/sqlfluff/core/templaters/slicers/tracer_plugins.py\\\\n@@ -65,6 +65,12 @@ class BlockPlugin:\\\\n         # Match jinja block start and end\\\\n-        match = re.match(r'({% (end)?block [^%]+ %})', template)\\\\n+        match = re.match(r'({% (end)?block [^%]+ %})', template, re.IGNORECASE)\\\\n+        if not match:\\\\n+            # If no match, return without modification\\\\n+            return template\\\\n+\\\\n+        block_start, block_end = match.span()\\\\n\\\\n         # Add handling for block tags specifically\\\\n         if match.group(2) is None:  # Start block\\\\n@@ -72,7 +78,8 @@ class BlockPlugin:\\\\n             # Adjust positions based on match and content\\\\n-            start_end_position = (len('{% block %}'), len('{% endblock %}'))\\\\n+            start_end_position = (\\\\n+                match.span(1)[0], match.span(1)[1]\\\\n+            )\\\\n             # Process start block\\\\n             return template[:block_start], block_name, start_end_position\\\\n         else:\\\\n         # Process end block\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch attempts to improve the matching of block tags but does not directly address the slicing issue. It is more of an enhancement to the existing logic rather than a fix for the reported problem.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective as it directly addresses the core issue with minimal risk. Patch 3 provides a useful fallback mechanism but is less precise. Patch 2 enhances block tag matching but does not directly solve the slicing problem.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    query = \\\\\\\"\\\\\\\"\\\\\\\"SELECT\\\\n\\\\t{% block table_name %}a{% endblock %}.b\\\\nFROM d.{{ self.table_name() }}\\\\\\\"\\\\\\\"\\\\\\\"\\\\n    result = lint_query(query, templater='jinja')\\\\n    assert 'AssertionError' in result or 'Length of templated file mismatch with final slice' in result\\\",\\n      \\\"description\\\": \\\"Reproduces the original issue by running the provided query with the Jinja templater\\\",\\n      \\\"expected_behavior\\\": \\\"Should trigger an AssertionError or a length mismatch warning\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Test with an empty query\\\\n    empty_query = \\\\\\\"\\\\\\\"\\\\n    result_empty = lint_query(empty_query, templater='jinja')\\\\n    assert 'AssertionError' not in result_empty and 'Length of templated file mismatch with final slice' not in result_empty\\\\n    \\\\n    # Test with a query that doesn't contain any templating blocks\\\\n    non_templated_query = \\\\\\\"SELECT * FROM table\\\\\\\"\\\\n    result_non_templated = lint_query(non_templated_query, templater='jinja')\\\\n    assert 'AssertionError' not in result_non_templated and 'Length of templated file mismatch with final slice' not in result_non_templated\\\",\\n      \\\"description\\\": \\\"Test edge cases by running queries that are empty and do not contain any templating blocks with the Jinja templater\\\",\\n      \\\"expected_behavior\\\": \\\"Should not trigger any errors\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    query = \\\\\\\"\\\\\\\"\\\\\\\"SELECT\\\\n\\\\t{% block table_name %}a{% endblock %}.b\\\\nFROM d.{{ self.table_name() }}\\\\\\\"\\\\\\\"\\\\\\\"\\\\n    result = lint_query(query, templater='jinja')\\\\n    assert 'AssertionError' not in result and 'Length of templated file mismatch with final slice' not in result\\\",\\n      \\\"description\\\": \\\"Validates that the patch fixes the incorrect slicing behavior in the Jinja templater\\\",\\n      \\\"expected_behavior\\\": \\\"Should not trigger any errors after applying the patches\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original issue, test edge cases, and validate the patches\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 251.066467, \"total_tokens\": 21519, \"total_steps\": 9, \"created_at\": 1753333638, \"finished_at\": 1753333889}}"
}