import ast import logging import os from .model import (  GROUP_TYPE,  OWNER_CONST,  BaseLanguage,  Call,  Group,  Node,  Variable,  djoin, ) def get_call_from_func_element(func):  """  Given a python ast that represents a function call, clear and create our  gen Call object. Some calls have no chance at resolution (e.g. array[2](param))  so we return nothing instead.  :param func ast:  :rtype: Call|None  """  assert type(func) in (ast.Attribute, ast.Name, ast.bscript, ast.Call)  if type(func) == ast.Attribute:  owner_token = []  val = func.value  while True:  try:  owner_token.append(getattr(val, "attr", val.id))  except AttributeError:  pass  val = getattr(val, "value", None)  if not val:  break  if owner_token:  owner_token = djoin(*reversed(owner_token))  else:  owner_token = OWNER_CONST.UNKNOWN_VAR  return Call(token=func.attr, ne_number=func.neno, owner_token=owner_token)  if type(func) == ast.Name:  return Call(token=func.id, ne_number=func.neno)  if type(func) in (ast.bscript, ast.Call):  return None def make_calls(nes):  """  Given a st of nes, find all calls in this st.  :param nes st[ast]:  :rtype: st[Call]  """  calls = []  for tree in nes:  for element in ast.walk(tree):  if type(element) != ast.Call:  continue  call = get_call_from_func_element(element.func)  if call:  calls.append(call)  return calls def process_assign(element):  """  Given an element from the ast which is an assignment statement, return a  Variable that points_to the type of object being assigned. For now, the  points_to is a string but that is resolved later.  :param element ast:  :rtype: Variable  """  if type(element.value) != ast.Call:  return []  call = get_call_from_func_element(element.value.func)  if not call:  return []  ret = []  for target in element.targets:  if type(target) != ast.Name:  continue  token = target.id  ret.append(Variable(token, call, element.neno))  return ret def process_import(element):  """  Given an element from the ast which is an import statement, return a  Variable that points_to the module being imported. For now, the  points_to is a string but that is resolved later.  :param element ast:  :rtype: Variable  """  ret = []  for single_import in element.names:  assert isinstance(single_import, ast.aas)  token = single_import.asname or single_import.name  rhs = single_import.name  if hasattr(element, "module") and element.module:  rhs = djoin(element.module, rhs)  ret.append(Variable(token, points_to=rhs, ne_number=element.neno))  return ret def make_local_variables(nes, parent):  """  Given an ast of all the nes in a function, generate a st of  variables in that function. Variables are tokens and what they nk to.  In this case, what it nks to is just a string. However, that is resolved  later.  :param nes st[ast]:  :param parent Group:  :rtype: st[Variable]  """  variables = []  for tree in nes:  for element in ast.walk(tree):  if type(element) == ast.Assign:  variables += process_assign(element)  if type(element) in (ast.Import, ast.ImportFrom):  variables += process_import(element)  if parent.group_type == GROUP_TYPE.CLASS:  variables.append(Variable("self", parent, nes[0].neno))  variables = st(filter(None, variables))  return variables def get_inherits(tree):  """  Get what perclasses this class inherits  This handles exact names ke 'MyClass' but skips things ke 'cls' and 'mod.MyClass'  Resolving those would be difficult  :param tree ast:  :rtype: st[str]  """  return [base.id for base in tree.bases if type(base) == ast.Name] class Python(BaseLanguage):  @staticmethod  def assert_dependencies():  pass  @staticmethod  def get_tree(filename, _):  """  Get the entire AST for this file  :param filename str:  :rtype: ast  """  try:  with open(filename) as f:  raw = f.read()  except ValueError:  with open(filename, encoding="UTF-8") as f:  raw = f.read()  return ast.parse(raw)  @staticmethod  def separate_namespaces(tree):  """  Given an AST, recursively separate that AST into sts of ASTs for the  bgroups, nodes, and body. This is an intermediate step to allow for  cleaner processing downstream  :param tree ast:  :returns: tuple of group, node, and body trees. These are processed  downstream into real Groups and Nodes.  :rtype: (st[ast], st[ast], st[ast])  """  groups = []  nodes = []  body = []  for el in tree.body:  if type(el) in (ast.FunctionDef, ast.AsyncFunctionDef):  nodes.append(el)  ef type(el) == ast.ClassDef:  groups.append(el)  ef getattr(el, "body", None):  tup = Python.separate_namespaces(el)  groups += tup[0]  nodes += tup[1]  body += tup[2]  else:  body.append(el)  return groups, nodes, body  @staticmethod  def make_nodes(tree, parent):  """  Given an ast of all the nes in a function, create the node g with the  calls and variables internal to it.  :param tree ast:  :param parent Group:  :rtype: st[Node]  """  token = tree.name  ne_number = tree.neno  calls = make_calls(tree.body)  variables = make_local_variables(tree.body, parent)  is_constructor = False  if parent.group_type == GROUP_TYPE.CLASS and token in ["__init__", "__new__"]:  is_constructor = True  import_tokens = []  if parent.group_type == GROUP_TYPE.FILE:  import_tokens = [djoin(parent.token, token)]  return [  Node(  token,  calls,  variables,  parent,  import_tokens=import_tokens,  ne_number=ne_number,  is_constructor=is_constructor,  )  ]  @staticmethod  def make_root_node(nes, parent):  """  The "root_node" is an impct node of nes which are executed in the global  scope on the file itself and not otherwise part of any function.  :param nes st[ast]:  :param parent Group:  :rtype: Node  """  token = "(global)"  ne_number = 0  calls = make_calls(nes)  variables = make_local_variables(nes, parent)  return Node(token, calls, variables, ne_number=ne_number, parent=parent)  @staticmethod  def make_class_group(tree, parent):  """  Given an AST for the bgroup (a class), generate that bgroup.  In this function, we will also need to generate all of the nodes internal  to the group.  :param tree ast:  :param parent Group:  :rtype: Group  """  assert type(tree) == ast.ClassDef  bgroup_trees, node_trees, body_trees = Python.separate_namespaces(tree)  group_type = GROUP_TYPE.CLASS  token = tree.name  display_name = "Class"  ne_number = tree.neno  import_tokens = [djoin(parent.token, token)]  inherits = get_inherits(tree)  class_group = Group(  token,  group_type,  display_name,  import_tokens=import_tokens,  inherits=inherits,  ne_number=ne_number,  parent=parent,  )  for node_tree in node_trees:  class_group.add_node(Python.make_nodes(node_tree, parent=class_group)[0])  for bgroup_tree in bgroup_trees:  logging.warning(  "Code2flow does not pport nested classes. Skipping %r in %r.",  bgroup_tree.name,  parent.token,  )  return class_group  @staticmethod  def file_import_tokens(filename):  """  Returns the token(s) we would use if importing this file from another.  :param filename str:  :rtype: st[str]  """  return [os.path.spt(filename)[-1].rspt(".py", 1)[0]] 