Skip to content
Snippets Groups Projects
definitions.py 6.39 KiB
Newer Older
  • Learn to ignore specific revisions
  • class DefinitionVisitor(ast.NodeVisitor):
        def __init__(self):
    
            self.functions = {}
            self.classes = {}
            self.names = {}
            self.attrs = set()
            self.definitions = {
    
    Amber Brown's avatar
    Amber Brown committed
                "def": self.functions,
                "class": self.classes,
                "names": self.names,
                "attrs": self.attrs,
    
            }
    
        def visit_Name(self, node):
            self.names.setdefault(type(node.ctx).__name__, set()).add(node.id)
    
        def visit_Attribute(self, node):
            self.attrs.add(node.attr)
            for child in ast.iter_child_nodes(node):
                self.visit(child)
    
        def visit_ClassDef(self, node):
            visitor = DefinitionVisitor()
            self.classes[node.name] = visitor.definitions
            for child in ast.iter_child_nodes(node):
                visitor.visit(child)
    
        def visit_FunctionDef(self, node):
            visitor = DefinitionVisitor()
            self.functions[node.name] = visitor.definitions
            for child in ast.iter_child_nodes(node):
                visitor.visit(child)
    
    
    def non_empty(defs):
    
    Amber Brown's avatar
    Amber Brown committed
        functions = {name: non_empty(f) for name, f in defs["def"].items()}
        classes = {name: non_empty(f) for name, f in defs["class"].items()}
    
    Amber Brown's avatar
    Amber Brown committed
            result["def"] = functions
    
    Amber Brown's avatar
    Amber Brown committed
            result["class"] = classes
        names = defs["names"]
    
    Amber Brown's avatar
    Amber Brown committed
        for name in names.get("Load", ()):
            if name not in names.get("Param", ()) and name not in names.get("Store", ()):
    
    Amber Brown's avatar
    Amber Brown committed
        uses.extend(defs["attrs"])
    
    Amber Brown's avatar
    Amber Brown committed
            result["uses"] = uses
        result["names"] = names
        result["attrs"] = defs["attrs"]
    
        return result
    
    
    def definitions_in_code(input_code):
        input_ast = ast.parse(input_code)
        visitor = DefinitionVisitor()
        visitor.visit(input_ast)
        definitions = non_empty(visitor.definitions)
        return definitions
    
    
    def definitions_in_file(filepath):
        with open(filepath) as f:
            return definitions_in_code(f.read())
    
    
    def defined_names(prefix, defs, names):
    
    Amber Brown's avatar
    Amber Brown committed
        for name, funcs in defs.get("def", {}).items():
            names.setdefault(name, {"defined": []})["defined"].append(prefix + name)
    
            defined_names(prefix + name + ".", funcs, names)
    
    
    Amber Brown's avatar
    Amber Brown committed
        for name, funcs in defs.get("class", {}).items():
            names.setdefault(name, {"defined": []})["defined"].append(prefix + name)
    
            defined_names(prefix + name + ".", funcs, names)
    
    
    
    def used_names(prefix, item, defs, names):
    
    Amber Brown's avatar
    Amber Brown committed
        for name, funcs in defs.get("def", {}).items():
    
            used_names(prefix + name + ".", name, funcs, names)
    
    Amber Brown's avatar
    Amber Brown committed
        for name, funcs in defs.get("class", {}).items():
    
            used_names(prefix + name + ".", name, funcs, names)
    
    Amber Brown's avatar
    Amber Brown committed
        path = prefix.rstrip(".")
        for used in defs.get("uses", ()):
    
    Amber Brown's avatar
    Amber Brown committed
                    names[item].setdefault("uses", []).append(used)
                names[used].setdefault("used", {}).setdefault(item, []).append(path)
    
    Amber Brown's avatar
    Amber Brown committed
    if __name__ == "__main__":
    
    Amber Brown's avatar
    Amber Brown committed
        parser = argparse.ArgumentParser(description="Find definitions.")
    
        parser.add_argument(
            "--unused", action="store_true", help="Only list unused definitions"
        )
        parser.add_argument(
            "--ignore", action="append", metavar="REGEXP", help="Ignore a pattern"
        )
        parser.add_argument(
    
            "--pattern", action="append", metavar="REGEXP", help="Search for a pattern"
    
        )
        parser.add_argument(
    
    Amber Brown's avatar
    Amber Brown committed
            nargs="+",
    
            metavar="DIR",
            help="Directories to search for definitions",
    
            "--referrers",
            default=0,
            type=int,
            help="Include referrers up to the given depth",
    
            "--referred",
            default=0,
            type=int,
            help="Include referred down to the given depth",
    
            "--format", default="yaml", help="Output format, one of 'yaml' or 'dot'"
    
        args = parser.parse_args()
    
        for directory in args.directories:
            for root, dirs, files in os.walk(directory):
                for filename in files:
                    if filename.endswith(".py"):
                        filepath = os.path.join(root, filename)
                        definitions[filepath] = definitions_in_file(filepath)
    
    
        names = {}
        for filepath, defs in definitions.items():
            defined_names(filepath + ":", defs, names)
    
        for filepath, defs in definitions.items():
    
            used_names(filepath + ":", None, defs, names)
    
        patterns = [re.compile(pattern) for pattern in args.pattern or ()]
        ignore = [re.compile(pattern) for pattern in args.ignore or ()]
    
        result = {}
        for name, definition in names.items():
            if patterns and not any(pattern.match(name) for pattern in patterns):
                continue
            if ignore and any(pattern.match(name) for pattern in ignore):
                continue
    
    Amber Brown's avatar
    Amber Brown committed
            if args.unused and definition.get("used"):
    
                continue
            result[name] = definition
    
    
        referrer_depth = args.referrers
        referrers = set()
        while referrer_depth:
            referrer_depth -= 1
            for entry in result.values():
    
    Mark Haines's avatar
    Mark Haines committed
                for used_by in entry.get("used", ()):
    
                    referrers.add(used_by)
            for name, definition in names.items():
    
                    continue
                if ignore and any(pattern.match(name) for pattern in ignore):
                    continue
                result[name] = definition
    
    
        referred_depth = args.referred
        referred = set()
        while referred_depth:
            referred_depth -= 1
            for entry in result.values():
                for uses in entry.get("uses", ()):
                    referred.add(uses)
            for name, definition in names.items():
    
                    continue
                if ignore and any(pattern.match(name) for pattern in ignore):
                    continue
                result[name] = definition
    
    
    Amber Brown's avatar
    Amber Brown committed
        if args.format == "yaml":
    
            yaml.dump(result, sys.stdout, default_flow_style=False)
    
    Amber Brown's avatar
    Amber Brown committed
        elif args.format == "dot":
    
            for name, entry in result.items():
    
                for used_by in entry.get("used", ()):
                    if used_by in result:
    
        else:
            raise ValueError("Unknown format %r" % (args.format))