Skip to content

isidentical/refactor

Repository files navigation

Refactor

PyPI version Documentation Try It

Simple, hassle-free, dependency-free, AST based fragmental source code refactoring and transformation toolkit.

Why?

Our framework is primarily built on the principle of "simple but effective transformations". We focus on refactorings that target a small span of source code, and work our way out from it. What this enables for us is being able to operate directly on a single format for both analyses and transformations. This is what we shine at compared to other similar tools.

How?

Let's not get into too much details, but just to give a sneak peek we can try to write a rule that would replace the identifier placeholder with 42.

import ast from refactor import Rule, Replace, run # Each refactor transformer inherits from "refactor.Rule" class FillPlaceholders(Rule): # And each rule implements a "match()" method, which would # receive every node in the tree in a breadth-first order. def match(self, node: ast.AST) -> Replace: # This is where things get interesting. Instead of just writing # filters with if statements, you can use the following assert # based approach (a contract of transformation). # For this case, our contract is going to be: if the given node # is an identifier with the name of "placeholder", it will be # replaced with literal "42". assert isinstance(node, ast.Name) assert node.id == "placeholder" # And this is where we choose what action we are taking for the # given node (which we have verified with our contract). There # are multiple transformation actions, but in this case what we # need is something that replaces a node with another one. replacement = ast.Constant(42) return Replace(node, replacement) if __name__ == "__main__": # And finally in here, we just use the default CLI that comes # bundled with refactor. When provided with a bunch of rules, # it creates a simple interface that can process given files # show the diff for changes and even apply them. run(rules=[FillPlaceholders])

If we run the rule above on a file, we can see how it performs:

--- test_file.py +++ test_file.py @@ -1,11 +1,11 @@ def main(): - print(placeholder * 3 + 2) + print(42 * 3 + 2) - print(2 + placeholder + 3) + print(2 + 42 + 3) # some comments - placeholder # maybe other comments + 42 # maybe other comments if something: other_thing - print(placeholder) + print(42) if __name__ == "__main__": main()

For learning more, check our documentation out!

About

AST-based fragmental source code refactoring toolkit for Python

Topics

Resources

License

Code of conduct

Stars

Watchers

Forks

Packages

No packages published

Languages