{
  "id": "sqlfluff__sqlfluff-3435",
  "question": "L027: outer-level table not found in WHERE clause sub-select\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\n\r\n### What Happened\r\n\r\nOuter-level table/view referenced in sub-select inside `WHERE` clause is not being detected.\r\n\r\nThis error seems to only occur when the sub-select contains joins.\r\n\r\n### Expected Behaviour\r\n\r\nNo error\r\n\r\n### Observed Behaviour\r\n\r\n```\r\nL:   7 | P:  32 | L027 | Qualified reference 'my_table.kind' not found in\r\n                       | available tables/view aliases ['other_table',\r\n                       | 'mapping_table'] in select with more than one referenced\r\n                       | table/view.\r\n```\r\n\r\n### How to reproduce\r\n\r\n```sql\r\nSELECT my_col\r\nFROM my_table\r\nWHERE EXISTS (\r\n    SELECT 1\r\n    FROM other_table\r\n    INNER JOIN mapping_table ON (mapping_table.other_fk = other_table.id_pk)\r\n    WHERE mapping_table.kind = my_table.kind\r\n);\r\n```\r\n\r\n### Dialect\r\n\r\npostgres\r\n\r\n### Version\r\n\r\nsqlfluff, version 0.12.0\r\n\r\n### Configuration\r\n\r\n```\r\n[sqlfluff]\r\nnocolor = True\r\ndialect = postgres\r\n```\r\n\r\n### Are you willing to work on and submit a PR to address the issue?\r\n\r\n- [ ] 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\": \"cd1ebd36-02c4-4f25-9ace-887ee0001396\", \"workflow_run_id\": \"c2392bae-ee95-4ee6-afcb-c61729acb96b\", \"data\": {\"id\": \"c2392bae-ee95-4ee6-afcb-c61729acb96b\", \"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/rules/L027.py b/src/sqlfluff/rules/L027.py\\\\nindex 7f8cbe4..b9e5d48 100644\\\\n--- a/src/sqlfluff/rules/L027.py\\\\n+++ b/src/sqlfluff/rules/L027.py\\\\n@@ -65,7 +65,12 @@ class Rule_L027(BaseRule):\\\\n                     + available_aliases\\\\n                 )\\\\n \\\\n-            available_aliases = set(available_aliases)\\\\n+            # Include outer context tables if available\\\\n+            if parent_stack:\\\\n+                outer_context_refs = parent_stack[0].get_context_references()\\\\n+                available_aliases = set(available_aliases).union(outer_context_refs)\\\\n+            else:\\\\n+                available_aliases = set(available_aliases)\\\\n \\\\n             # A self contained select and its references should exist.\\\\n             if current_refs and available_aliases:\\\\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 including outer context tables when determining available aliases. It does so with minimal changes to the existing logic, maintaining code quality and reducing the risk of introducing new bugs.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/rules/L027.py b/src/sqlfluff/rules/L027.py\\\\nindex 7f8cbe4..e081d50 100644\\\\n--- a/src/sqlfluff/rules/L027.py\\\\n+++ b/src/sqlfluff/rules/L027.py\\\\n@@ -67,6 +67,12 @@ class Rule_L027(BaseRule):\\\\n                     available_aliases\\\\n                 )\\\\n \\\\n+            # Check parent stack for outer context\\\\_tables\\\\n+            outer_context_aliases = set()\\\\n+            for parent in parent_stack:\\\\n+                outer_context_aliases.update(parent.get_context_references())\\\\n+\\\\n+            available_aliases.update(outer_context_aliases)\\\\n \\\\n             # A self contained select and its references should exist.\\\\n             if current_refs and available_aliases:\\\\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 also addresses the issue by iterating through the parent stack to gather outer context references. While effective, it introduces slightly more complexity and potential for errors due to the iteration over the parent stack.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/src/sqlfluff/utils/analysis/select.py b/src/sqlfluff/utils/analysis/select.py\\\\nindex f12c3b8..cd34f10 100644\\\\n--- a/src/sqlfluff/utils/analysis/select.py\\\\n+++ b/src/sqlfluff/utils/analysis/select.py\\\\n@@ -135,6 +135,14 @@ def get_select_statement_info(segment):\\\\n     )\\\\n \\\\n def propagate_outer_context(segment, parent_stack):\\\\n+    \\\\\\\"\\\\\\\"\\\\\\\"Adds outer-level context tables to subqueries.\\\\\\\"\\\\\\\"\\\\\\\"\\\\n+    if parent_stack:\\\\n+        outer_context_refs = parent_stack[0].get_context_references()\\\\n+        outer_tables = {t for t in outer_context_refs if t not in segment.tables}\\\\n+        segment.tables.update(outer_tables)\\\\n+    return segment\\\\n+\\\\n+\\\\n def analyze_segment(segment, parent_stack=None):\\\\n     \\\\\\\"\\\\\\\"\\\\\\\"Analyze tables and references within a segment, considering any outer context.\\\\\\\"\\\\\\\"\\\\\\\"\\\\n     if parent_stack:\\\\n@@ -145,6 +153,7 @@ def analyze_segment(segment, parent_stack=None):\\\\n             return segment\\\\n         \\\\n         # Update segment analysis\\\\n+        segment = propagate_outer_context(segment, parent_stack)\\\\n         select_info = get_select_statement_info(segment)\\\\n         # Do more analysis...\\\\n         return select_info\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch introduces a new utility function to handle outer context propagation. While it provides a comprehensive solution, it adds more complexity and changes to the codebase, increasing the risk of unintended side effects.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective and least risky solution, making minimal changes to the existing logic while addressing the issue. Patch 2 is a viable alternative but introduces more complexity. Patch 3, while comprehensive, adds significant complexity and potential for side effects, making it the least preferred option.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_with_join\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_with_join():\\\\n    sql_query = '''\\\\n        SELECT my_col\\\\n        FROM my_table\\\\n        WHERE EXISTS (\\\\n            SELECT 1\\\\n            FROM other_table\\\\n            INNER JOIN mapping_table ON (mapping_table.other_fk = other_table.id_pk)\\\\n            WHERE mapping_table.kind = my_table.kind\\\\n        );\\\\n    '''\\\\n    result = analyze_sql_query(sql_query)\\\\n    assert 'L027' in result\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue with a SQL query containing a join in the sub-select inside the WHERE clause\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail before applying the patches due to the L027 error\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_case_no_join\\\",\\n      \\\"test_code\\\": \\\"def test_edge_case_no_join():\\\\n    sql_query = '''\\\\n        SELECT my_col\\\\n        FROM my_table\\\\n        WHERE EXISTS (\\\\n            SELECT 1\\\\n            FROM other_table\\\\n            WHERE other_table.id_pk = my_table.id_pk\\\\n        );\\\\n    '''\\\\n    result = analyze_sql_query(sql_query)\\\\n    assert 'L027' not in result\\\\n\\\",\\n      \\\"description\\\": \\\"This test covers an edge case without a join in the sub-select inside the WHERE clause\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass showing that the issue does not occur in this scenario\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_1_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_1_validation():\\\\n    sql_query = '''\\\\n        SELECT my_col\\\\n        FROM my_table\\\\n        WHERE EXISTS (\\\\n            SELECT 1\\\\n            FROM other_table\\\\n            INNER JOIN mapping_table ON (mapping_table.other_fk = other_table.id_pk)\\\\n            WHERE mapping_table.kind = my_table.kind\\\\n        );\\\\n    '''\\\\n    result = analyze_sql_query(sql_query)\\\\n    assert 'L027' not in result\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates the first patch proposal by checking if the L027 error is resolved\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass after applying patch 1\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_2_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_2_validation():\\\\n    sql_query = '''\\\\n        SELECT my_col\\\\n        FROM my_table\\\\n        WHERE EXISTS (\\\\n            SELECT 1\\\\n            FROM other_table\\\\n            INNER JOIN mapping_table ON (mapping_table.other_fk = other_table.id_pk)\\\\n            WHERE mapping_table.kind = my_table.kind\\\\n        );\\\\n    '''\\\\n    result = analyze_sql_query(sql_query)\\\\n    assert 'L027' not in result\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates the second patch proposal by checking if the L027 error is resolved\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass after applying patch 2\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_3_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_3_validation():\\\\n    sql_query = '''\\\\n        SELECT my_col\\\\n        FROM my_table\\\\n        WHERE EXISTS (\\\\n            SELECT 1\\\\n            FROM other_table\\\\n            INNER JOIN mapping_table ON (mapping_table.other_fk = other_table.id_pk)\\\\n            WHERE mapping_table.kind = my_table.kind\\\\n        );\\\\n    '''\\\\n    result = analyze_sql_query(sql_query)\\\\n    assert 'L027' not in result\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates the third patch proposal by checking if the L027 error is resolved\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass after applying patch 3\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases have been generated to reproduce the original issue, cover edge cases, and validate the proposed patches for resolving the L027 error.\\\"\\n}\\n```  \"}, \"error\": \"\", \"elapsed_time\": 300.275679, \"total_tokens\": 22367, \"total_steps\": 9, \"created_at\": 1753335536, \"finished_at\": 1753335836}}"
}