File: _gather_global_names.py

package info (click to toggle)
python-libcst 1.4.0-1.2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 5,928 kB
  • sloc: python: 76,235; makefile: 10; sh: 2
file content (75 lines) | stat: -rw-r--r-- 2,836 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
# Copyright (c) Meta Platforms, Inc. and affiliates.
#
# This source code is licensed under the MIT license found in the
# LICENSE file in the root directory of this source tree.

from typing import Set

import libcst
from libcst.codemod._context import CodemodContext
from libcst.codemod._visitor import ContextAwareVisitor


class GatherGlobalNamesVisitor(ContextAwareVisitor):
    """
    Gathers all globally accessible names defined in a module and stores them as
    attributes on the instance.
    Intended to be instantiated and passed to a :class:`~libcst.Module`
    :meth:`~libcst.CSTNode.visit` method in order to gather up information about
    names defined on a module. Note that this is not a substitute for scope
    analysis or qualified name support. Please see :ref:`libcst-scope-tutorial`
    for a more robust way of determining the qualified name and definition for
    an arbitrary node.
    Names that are globally accessible through imports are currently not included
    but can be retrieved with GatherImportsVisitor.

    After visiting a module the following attributes will be populated:

     global_names
      A sequence of strings representing global variables defined in the module
      toplevel.
     class_names
      A sequence of strings representing classes defined in the module toplevel.
     function_names
      A sequence of strings representing functions defined in the module toplevel.

    """

    def __init__(self, context: CodemodContext) -> None:
        super().__init__(context)
        self.global_names: Set[str] = set()
        self.class_names: Set[str] = set()
        self.function_names: Set[str] = set()
        # Track scope nesting
        self.scope_depth: int = 0

    def visit_ClassDef(self, node: libcst.ClassDef) -> None:
        if self.scope_depth == 0:
            self.class_names.add(node.name.value)
        self.scope_depth += 1

    def leave_ClassDef(self, original_node: libcst.ClassDef) -> None:
        self.scope_depth -= 1

    def visit_FunctionDef(self, node: libcst.FunctionDef) -> None:
        if self.scope_depth == 0:
            self.function_names.add(node.name.value)
        self.scope_depth += 1

    def leave_FunctionDef(self, original_node: libcst.FunctionDef) -> None:
        self.scope_depth -= 1

    def visit_Assign(self, node: libcst.Assign) -> None:
        if self.scope_depth != 0:
            return
        for assign_target in node.targets:
            target = assign_target.target
            if isinstance(target, libcst.Name):
                self.global_names.add(target.value)

    def visit_AnnAssign(self, node: libcst.AnnAssign) -> None:
        if self.scope_depth != 0:
            return
        target = node.target
        if isinstance(target, libcst.Name):
            self.global_names.add(target.value)